##// END OF EJS Templates
py3: use r'' instead of sysstr('') to get around code transformer...
py3: use r'' instead of sysstr('') to get around code transformer Fewer function calls should be better.

File last commit:

r36729:e79adc12 default
r36853:5bc7ff10 default
Show More
templater.py
1642 lines | 57.6 KiB | text/x-python | PythonLexer
Vadim Gelfer
add doc comments to template code.
r1909 # templater.py - template expansion for output
#
# Copyright 2005, 2006 Matt Mackall <mpm@selenic.com>
#
Martin Geisler
updated license to be explicit about GPL version 2
r8225 # 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.
Vadim Gelfer
add doc comments to template code.
r1909
Yuya Nishihara
doctest: use print_function and convert bytes to unicode where needed
r34139 from __future__ import absolute_import, print_function
Gregory Szorc
templater: use absolute_import
r25985
import os
import re
Weiwen
template engine: convert generator-based iterator to list-based iterator...
r17982 import types
Gregory Szorc
templater: use absolute_import
r25985
from .i18n import _
from . import (
Yuya Nishihara
templater: make pad() strip color codes before computing width (issue5416)...
r31521 color,
Gregory Szorc
templater: use absolute_import
r25985 config,
Yuya Nishihara
templater: make pad() compute actual width...
r31520 encoding,
Gregory Szorc
templater: use absolute_import
r25985 error,
minirst,
Boris Feld
template: compute verb in obsfateverb...
r33995 obsutil,
Gregory Szorc
templater: use absolute_import
r25985 parser,
Pulkit Goyal
py3: replace os.sep with pycompat.ossep (part 3 of 4)
r30615 pycompat,
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 registrar,
Gregory Szorc
templater: use absolute_import
r25985 revset as revsetmod,
Yuya Nishihara
revset: split language services to revsetlang module (API)...
r31024 revsetlang,
Yuya Nishihara
templater: add experimental support for extdata...
r34459 scmutil,
Gregory Szorc
templater: use absolute_import
r25985 templatefilters,
templatekw,
util,
)
Boris Feld
util: extract all date-related utils in utils/dateutil module...
r36625 from .utils import dateutil
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176
Yuya Nishihara
templater: specialize ResourceUnavailable error so that it can be caught...
r36462 class ResourceUnavailable(error.Abort):
pass
Yuya Nishihara
templater: move specialized exception types to top...
r36461 class TemplateNotFound(error.Abort):
pass
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 # template parsing
elements = {
Yuya Nishihara
parser: separate actions for primary expression and prefix operator...
r25815 # token-type: binding-strength, primary, prefix, infix, suffix
"(": (20, None, ("group", 1, ")"), ("func", 1, ")"), None),
Yuya Nishihara
templater: add dot operator to easily access a sub item...
r34536 ".": (18, None, None, (".", 18), None),
Yuya Nishihara
templater: adjust binding strength of '%' and '|' operators (BC)...
r34330 "%": (15, None, None, ("%", 15), None),
Yuya Nishihara
templater: adjust binding strengths to make room for key-value operator...
r31884 "|": (15, None, None, ("|", 15), None),
"*": (5, None, None, ("*", 5), None),
"/": (5, None, None, ("/", 5), None),
"+": (4, None, None, ("+", 4), None),
"-": (4, None, ("negate", 19), ("-", 4), None),
Yuya Nishihara
templater: add parsing rule for key-value pair...
r31885 "=": (3, None, None, ("keyvalue", 3), None),
Yuya Nishihara
templater: sort token table by binding strength...
r31883 ",": (2, None, None, ("list", 2), None),
")": (0, None, None, None, None),
Yuya Nishihara
parser: separate actions for primary expression and prefix operator...
r25815 "integer": (0, "integer", None, None, None),
"symbol": (0, "symbol", None, None, None),
"string": (0, "string", None, None, None),
"template": (0, "template", None, None, None),
"end": (0, None, None, None, None),
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 }
Yuya Nishihara
templater: add function to parse whole string as template expression...
r28911 def tokenize(program, start, end, term=None):
"""Parse a template expression into a stream of tokens, which must end
with term if specified"""
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 pos = start
Pulkit Goyal
py3: use pycompat.bytestr instead of bytes
r32154 program = pycompat.bytestr(program)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 while pos < end:
c = program[pos]
if c.isspace(): # skip inter-token whitespace
pass
Yuya Nishihara
templater: add dot operator to easily access a sub item...
r34536 elif c in "(=,).%|+-*/": # handle simple operators
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 yield (c, None, pos)
Yuya Nishihara
templater: introduce one-pass parsing of nested template strings...
r25783 elif c in '"\'': # handle quoted templates
s = pos + 1
data, pos = _parsetemplate(program, s, end, c)
yield ('template', data, s)
pos -= 1
Yuya Nishihara
templater: remove processing of "string" literals from tokenizer...
r25784 elif c == 'r' and program[pos:pos + 2] in ("r'", 'r"'):
# handle quoted strings
c = program[pos + 1]
s = pos = pos + 2
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 while pos < end: # find closing quote
d = program[pos]
Yuya Nishihara
templater: fix handling of \-escapes in raw string literals...
r25638 if d == '\\': # skip over escaped characters
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 pos += 2
continue
if d == c:
Yuya Nishihara
templater: unify "string" and "rawstring"...
r25785 yield ('string', program[s:pos], s)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 break
pos += 1
else:
raise error.ParseError(_("unterminated string"), s)
Simon Farnsworth
templater: provide arithmetic operations on integers...
r30115 elif c.isdigit():
Yuya Nishihara
templater: tokenize decimal integer literal (issue4638) (BC)...
r25002 s = pos
while pos < end:
d = program[pos]
if not d.isdigit():
break
pos += 1
yield ('integer', program[s:pos], s)
pos -= 1
Yuya Nishihara
py3: fix type of string literals in templater.tokenize()...
r36564 elif (c == '\\' and program[pos:pos + 2] in (br"\'", br'\"')
or c == 'r' and program[pos:pos + 3] in (br"r\'", br'r\"')):
Yuya Nishihara
templater: parse \"...\" as string for 2.9.2-3.4 compatibility (issue4733)...
r25676 # handle escaped quoted strings for compatibility with 2.9.2-3.4,
# where some of nested templates were preprocessed as strings and
# then compiled. therefore, \"...\" was allowed. (issue4733)
#
# processing flow of _evalifliteral() at 5ab28a2e9962:
# outer template string -> stringify() -> compiletemplate()
# ------------------------ ------------ ------------------
# {f("\\\\ {g(\"\\\"\")}"} \\ {g("\"")} [r'\\', {g("\"")}]
# ~~~~~~~~
# escaped quoted string
if c == 'r':
pos += 1
Yuya Nishihara
templater: unify "string" and "rawstring"...
r25785 token = 'string'
Yuya Nishihara
templater: parse \"...\" as string for 2.9.2-3.4 compatibility (issue4733)...
r25676 else:
Yuya Nishihara
templater: introduce one-pass parsing of nested template strings...
r25783 token = 'template'
Yuya Nishihara
templater: parse \"...\" as string for 2.9.2-3.4 compatibility (issue4733)...
r25676 quote = program[pos:pos + 2]
s = pos = pos + 2
while pos < end: # find closing escaped quote
if program.startswith('\\\\\\', pos, end):
pos += 4 # skip over double escaped characters
continue
if program.startswith(quote, pos, end):
Matt Mackall
templater: create string unescape helper (issue4798)...
r26215 # interpret as if it were a part of an outer string
Yuya Nishihara
parser: move unescape helper from templater...
r26231 data = parser.unescapestr(program[s:pos])
Yuya Nishihara
templater: introduce one-pass parsing of nested template strings...
r25783 if token == 'template':
data = _parsetemplate(data, 0, len(data))[0]
Yuya Nishihara
templater: parse \"...\" as string for 2.9.2-3.4 compatibility (issue4733)...
r25676 yield (token, data, s)
pos += 1
break
pos += 1
else:
raise error.ParseError(_("unterminated string"), s)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 elif c.isalnum() or c in '_':
s = pos
pos += 1
while pos < end: # find end of symbol
d = program[pos]
if not (d.isalnum() or d == "_"):
break
pos += 1
sym = program[s:pos]
Brendan Cully
templater: back out 0615b22da148, it breaks schemes ({1})
r18893 yield ('symbol', sym, s)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 pos -= 1
Yuya Nishihara
templater: add function to parse whole string as template expression...
r28911 elif c == term:
Yuya Nishihara
templater: fix position of terminator character in error message...
r36709 yield ('end', None, pos)
Yuya Nishihara
templater: check existence of closing brace of template string
r25782 return
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 else:
raise error.ParseError(_("syntax error"), pos)
pos += 1
Yuya Nishihara
templater: add function to parse whole string as template expression...
r28911 if term:
raise error.ParseError(_("unterminated template expansion"), start)
yield ('end', None, pos)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176
Yuya Nishihara
templater: introduce one-pass parsing of nested template strings...
r25783 def _parsetemplate(tmpl, start, stop, quote=''):
r"""
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _parsetemplate(b'foo{bar}"baz', 0, 12)
Yuya Nishihara
templater: introduce one-pass parsing of nested template strings...
r25783 ([('string', 'foo'), ('symbol', 'bar'), ('string', '"baz')], 12)
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _parsetemplate(b'foo{bar}"baz', 0, 12, quote=b'"')
Yuya Nishihara
templater: introduce one-pass parsing of nested template strings...
r25783 ([('string', 'foo'), ('symbol', 'bar')], 9)
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _parsetemplate(b'foo"{bar}', 0, 9, quote=b'"')
Yuya Nishihara
templater: introduce one-pass parsing of nested template strings...
r25783 ([('string', 'foo')], 4)
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _parsetemplate(br'foo\"bar"baz', 0, 12, quote=b'"')
Yuya Nishihara
templater: introduce one-pass parsing of nested template strings...
r25783 ([('string', 'foo"'), ('string', 'bar')], 9)
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _parsetemplate(br'foo\\"bar', 0, 10, quote=b'"')
Yuya Nishihara
templater: unify "string" and "rawstring"...
r25785 ([('string', 'foo\\')], 6)
Yuya Nishihara
templater: introduce one-pass parsing of nested template strings...
r25783 """
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 parsed = []
Yuya Nishihara
templater: extract function scanning template string...
r36258 for typ, val, pos in _scantemplate(tmpl, start, stop, quote):
if typ == 'string':
parsed.append((typ, val))
elif typ == 'template':
parsed.append(val)
elif typ == 'end':
return parsed, pos
else:
raise error.ProgrammingError('unexpected type: %s' % typ)
raise error.ProgrammingError('unterminated scanning of template')
Yuya Nishihara
templater: add option to parse template string just like raw string literal...
r36527 def scantemplate(tmpl, raw=False):
r"""Scan (type, start, end) positions of outermost elements in template
If raw=True, a backslash is not taken as an escape character just like
r'' string in Python. Note that this is different from r'' literal in
template in that no template fragment can appear in r'', e.g. r'{foo}'
is a literal '{foo}', but ('{foo}', raw=True) is a template expression
'foo'.
Yuya Nishihara
templater: add function to help substituting patterns in template string...
r36259
>>> list(scantemplate(b'foo{bar}"baz'))
[('string', 0, 3), ('template', 3, 8), ('string', 8, 12)]
>>> list(scantemplate(b'outer{"inner"}outer'))
[('string', 0, 5), ('template', 5, 14), ('string', 14, 19)]
>>> list(scantemplate(b'foo\\{escaped}'))
[('string', 0, 5), ('string', 5, 13)]
Yuya Nishihara
templater: add option to parse template string just like raw string literal...
r36527 >>> list(scantemplate(b'foo\\{escaped}', raw=True))
[('string', 0, 4), ('template', 4, 13)]
Yuya Nishihara
templater: add function to help substituting patterns in template string...
r36259 """
last = None
Yuya Nishihara
templater: add option to parse template string just like raw string literal...
r36527 for typ, val, pos in _scantemplate(tmpl, 0, len(tmpl), raw=raw):
Yuya Nishihara
templater: add function to help substituting patterns in template string...
r36259 if last:
yield last + (pos,)
if typ == 'end':
return
else:
last = (typ, pos)
raise error.ProgrammingError('unterminated scanning of template')
Yuya Nishihara
templater: add option to parse template string just like raw string literal...
r36527 def _scantemplate(tmpl, start, stop, quote='', raw=False):
Yuya Nishihara
templater: extract function scanning template string...
r36258 """Parse template string into chunks of strings and template expressions"""
Yuya Nishihara
templater: introduce one-pass parsing of nested template strings...
r25783 sepchars = '{' + quote
Yuya Nishihara
templater: add option to parse template string just like raw string literal...
r36527 unescape = [parser.unescapestr, pycompat.identity][raw]
Yuya Nishihara
templater: extract function that parses template string...
r25781 pos = start
Yuya Nishihara
parser: accept iterator of tokens instead of tokenizer function and program...
r25654 p = parser.parser(elements)
Ryan McElroy
templater: add hint to template parse errors to help locate issues...
r36687 try:
while pos < stop:
n = min((tmpl.find(c, pos, stop) for c in sepchars),
key=lambda n: (n < 0, n))
if n < 0:
yield ('string', unescape(tmpl[pos:stop]), pos)
pos = stop
break
c = tmpl[n:n + 1]
bs = 0 # count leading backslashes
if not raw:
bs = (n - pos) - len(tmpl[pos:n].rstrip('\\'))
if bs % 2 == 1:
# escaped (e.g. '\{', '\\\{', but not '\\{')
yield ('string', unescape(tmpl[pos:n - 1]) + c, pos)
pos = n + 1
continue
if n > pos:
yield ('string', unescape(tmpl[pos:n]), pos)
if c == quote:
yield ('end', None, n + 1)
return
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176
Ryan McElroy
templater: add hint to template parse errors to help locate issues...
r36687 parseres, pos = p.parse(tokenize(tmpl, n + 1, stop, '}'))
Yuya Nishihara
templater: fix position of terminator character in error message...
r36709 if not tmpl.startswith('}', pos):
Ryan McElroy
templater: add hint to template parse errors to help locate issues...
r36687 raise error.ParseError(_("invalid token"), pos)
yield ('template', parseres, n)
Yuya Nishihara
templater: fix position of terminator character in error message...
r36709 pos += 1
Yuya Nishihara
templater: introduce one-pass parsing of nested template strings...
r25783
Ryan McElroy
templater: add hint to template parse errors to help locate issues...
r36687 if quote:
raise error.ParseError(_("unterminated string"), start)
except error.ParseError as inst:
if len(inst.args) > 1: # has location
loc = inst.args[1]
Ryan McElroy
templater: provide hint for multi-line templates with parse errors...
r36688 # Offset the caret location by the number of newlines before the
# location of the error, since we will replace one-char newlines
# with the two-char literal r'\n'.
offset = tmpl[:loc].count('\n')
tmpl = tmpl.replace('\n', br'\n')
# We want the caret to point to the place in the template that
# failed to parse, but in a hint we get a open paren at the
Yuya Nishihara
templater: fix position of terminator character in error message...
r36709 # start. Therefore, we print "loc + 1" spaces (instead of "loc")
Ryan McElroy
templater: provide hint for multi-line templates with parse errors...
r36688 # to line up the caret with the location of the error.
Yuya Nishihara
templater: fix position of terminator character in error message...
r36709 inst.hint = (tmpl + '\n'
+ ' ' * (loc + 1 + offset) + '^ ' + _('here'))
Ryan McElroy
templater: add hint to template parse errors to help locate issues...
r36687 raise
Yuya Nishihara
templater: extract function scanning template string...
r36258 yield ('end', None, pos)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176
Yuya Nishihara
templater: expand list of parsed templates to template node...
r28547 def _unnesttemplatelist(tree):
"""Expand list of templates to node tuple
>>> def f(tree):
Yuya Nishihara
doctest: use print_function and convert bytes to unicode where needed
r34139 ... print(pycompat.sysstr(prettyformat(_unnesttemplatelist(tree))))
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> f((b'template', []))
Yuya Nishihara
parser: stabilize output of prettyformat() by using byte-safe repr()...
r34075 (string '')
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> f((b'template', [(b'string', b'foo')]))
Yuya Nishihara
parser: stabilize output of prettyformat() by using byte-safe repr()...
r34075 (string 'foo')
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> f((b'template', [(b'string', b'foo'), (b'symbol', b'rev')]))
Yuya Nishihara
templater: expand list of parsed templates to template node...
r28547 (template
Yuya Nishihara
parser: stabilize output of prettyformat() by using byte-safe repr()...
r34075 (string 'foo')
(symbol 'rev'))
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> f((b'template', [(b'symbol', b'rev')])) # template(rev) -> str
Yuya Nishihara
templater: expand list of parsed templates to template node...
r28547 (template
Yuya Nishihara
parser: stabilize output of prettyformat() by using byte-safe repr()...
r34075 (symbol 'rev'))
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> f((b'template', [(b'template', [(b'string', b'foo')])]))
Yuya Nishihara
parser: stabilize output of prettyformat() by using byte-safe repr()...
r34075 (string 'foo')
Yuya Nishihara
templater: expand list of parsed templates to template node...
r28547 """
if not isinstance(tree, tuple):
return tree
op = tree[0]
if op != 'template':
return (op,) + tuple(_unnesttemplatelist(x) for x in tree[1:])
assert len(tree) == 2
xs = tuple(_unnesttemplatelist(x) for x in tree[1])
if not xs:
return ('string', '') # empty template ""
elif len(xs) == 1 and xs[0][0] == 'string':
return xs[0] # fast path for string with no template fragment "x"
else:
return (op,) + xs
Yuya Nishihara
templater: lift parsed and compiled templates to generic data types...
r28545 def parse(tmpl):
"""Parse template string into tree"""
Yuya Nishihara
templater: extract function that parses template string...
r25781 parsed, pos = _parsetemplate(tmpl, 0, len(tmpl))
Yuya Nishihara
templater: lift parsed and compiled templates to generic data types...
r28545 assert pos == len(tmpl), 'unquoted template should be consumed'
Yuya Nishihara
templater: expand list of parsed templates to template node...
r28547 return _unnesttemplatelist(('template', parsed))
Yuya Nishihara
templater: add function to parse whole string as template expression...
r28911 def _parseexpr(expr):
"""Parse a template expression into tree
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _parseexpr(b'"foo"')
Yuya Nishihara
templater: add function to parse whole string as template expression...
r28911 ('string', 'foo')
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _parseexpr(b'foo(bar)')
Yuya Nishihara
templater: add function to parse whole string as template expression...
r28911 ('func', ('symbol', 'foo'), ('symbol', 'bar'))
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _parseexpr(b'foo(')
Yuya Nishihara
templater: add function to parse whole string as template expression...
r28911 Traceback (most recent call last):
...
ParseError: ('not a prefix: end', 4)
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> _parseexpr(b'"foo" "bar"')
Yuya Nishihara
templater: add function to parse whole string as template expression...
r28911 Traceback (most recent call last):
...
ParseError: ('invalid token', 7)
"""
p = parser.parser(elements)
tree, pos = p.parse(tokenize(expr, 0, len(expr)))
if pos != len(expr):
raise error.ParseError(_('invalid token'), pos)
return _unnesttemplatelist(tree)
Yuya Nishihara
templater: expand list of parsed templates to template node...
r28547 def prettyformat(tree):
return parser.prettyformat(tree, ('integer', 'string', 'symbol'))
Yuya Nishihara
templater: lift parsed and compiled templates to generic data types...
r28545
Yuya Nishihara
templater: switch methods table on compileexp() of func args and inner expr...
r25001 def compileexp(exp, context, curmethods):
Yuya Nishihara
templater: inline compiletemplate() function into engine...
r28956 """Compile parsed template tree to (func, data) pair"""
Yuya Nishihara
templater: fix crash by empty group expression...
r35762 if not exp:
raise error.ParseError(_("missing argument"))
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 t = exp[0]
Yuya Nishihara
templater: switch methods table on compileexp() of func args and inner expr...
r25001 if t in curmethods:
return curmethods[t](exp, context)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 raise error.ParseError(_("unknown method '%s'") % t)
# template evaluation
def getsymbol(exp):
if exp[0] == 'symbol':
return exp[1]
Ryan McElroy
templater: add symbol to error...
r21822 raise error.ParseError(_("expected a symbol, got '%s'") % exp[0])
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176
def getlist(x):
if not x:
return []
if x[0] == 'list':
return getlist(x[1]) + [x[2]]
return [x]
def gettemplate(exp, context):
Yuya Nishihara
templater: lift parsed and compiled templates to generic data types...
r28545 """Compile given template tree or load named template from map file;
returns (func, data) pair"""
Yuya Nishihara
templater: relax type of mapped template...
r28546 if exp[0] in ('template', 'string'):
Yuya Nishihara
templater: lift parsed and compiled templates to generic data types...
r28545 return compileexp(exp, context, methods)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 if exp[0] == 'symbol':
Yuya Nishihara
templater: comment that gettemplate() has different name resolution order...
r25599 # unlike runsymbol(), here 'symbol' is always taken as template name
# even if it exists in mapping. this allows us to override mapping
# by web templates, e.g. 'changelogtag' is redefined in map file.
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 return context._load(exp[1])
raise error.ParseError(_("expected template specifier"))
Yuya Nishihara
templater: find keyword name more thoroughly on filtering error...
r31927 def findsymbolicname(arg):
"""Find symbolic name for the given compiled expression; returns None
if nothing found reliably"""
while True:
func, data = arg
if func is runsymbol:
return data
elif func is runfilter:
arg = data[0]
else:
return None
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 def evalrawexp(context, mapping, arg):
"""Evaluate given argument as a bare template object which may require
further processing (such as folding generator of strings)"""
Yuya Nishihara
templater: extract helper that evaluates filter or function argument...
r26124 func, data = arg
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 return func(context, mapping, data)
def evalfuncarg(context, mapping, arg):
"""Evaluate given argument as value type"""
thing = evalrawexp(context, mapping, arg)
Yuya Nishihara
templatekw: add new-style template expansion to {manifest}...
r34331 thing = templatekw.unwrapvalue(thing)
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 # evalrawexp() may return string, generator of strings or arbitrary object
# such as date tuple, but filter does not want generator.
Yuya Nishihara
templater: extract helper that evaluates filter or function argument...
r26124 if isinstance(thing, types.GeneratorType):
thing = stringify(thing)
return thing
Yuya Nishihara
templater: fix if() to not evaluate False as bool('False')...
r29816 def evalboolean(context, mapping, arg):
Yuya Nishihara
templater: make pad() evaluate boolean argument (BC)...
r29817 """Evaluate given argument as boolean, but also takes boolean literals"""
Yuya Nishihara
templater: fix if() to not evaluate False as bool('False')...
r29816 func, data = arg
Yuya Nishihara
templater: make pad() evaluate boolean argument (BC)...
r29817 if func is runsymbol:
thing = func(context, mapping, data, default=None)
if thing is None:
# not a template keyword, takes as a boolean literal
thing = util.parsebool(data)
else:
thing = func(context, mapping, data)
Yuya Nishihara
templatekw: add new-style template expansion to {manifest}...
r34331 thing = templatekw.unwrapvalue(thing)
Yuya Nishihara
templater: fix if() to not evaluate False as bool('False')...
r29816 if isinstance(thing, bool):
return thing
# other objects are evaluated as strings, which means 0 is True, but
# empty dict/list should be False as they are expected to be ''
return bool(stringify(thing))
Yuya Nishihara
templater: store revisions as ints so min/max won't compare them as strings...
r34582 def evalinteger(context, mapping, arg, err=None):
Yuya Nishihara
templater: drop redundant type conversion when evaluating integer argument...
r28344 v = evalfuncarg(context, mapping, arg)
Yuya Nishihara
templater: factor out function that evaluates argument as integer...
r28343 try:
Yuya Nishihara
templater: drop redundant type conversion when evaluating integer argument...
r28344 return int(v)
except (TypeError, ValueError):
Yuya Nishihara
templater: store revisions as ints so min/max won't compare them as strings...
r34582 raise error.ParseError(err or _('not an integer'))
Yuya Nishihara
templater: factor out function that evaluates argument as integer...
r28343
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 def evalstring(context, mapping, arg):
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 return stringify(evalrawexp(context, mapping, arg))
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348
Yuya Nishihara
templater: make label() take unknown symbol as color literal...
r28373 def evalstringliteral(context, mapping, arg):
"""Evaluate given argument as string template, but returns symbol name
if it is unknown"""
func, data = arg
if func is runsymbol:
thing = func(context, mapping, data, default=data)
else:
thing = func(context, mapping, data)
return stringify(thing)
Yuya Nishihara
templater: store revisions as ints so min/max won't compare them as strings...
r34582 _evalfuncbytype = {
bool: evalboolean,
bytes: evalstring,
int: evalinteger,
}
def evalastype(context, mapping, arg, typ):
"""Evaluate given argument and coerce its type"""
try:
f = _evalfuncbytype[typ]
except KeyError:
raise error.ProgrammingError('invalid type specified: %r' % typ)
return f(context, mapping, arg)
Yuya Nishihara
templater: tokenize decimal integer literal (issue4638) (BC)...
r25002 def runinteger(context, mapping, data):
return int(data)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 def runstring(context, mapping, data):
return data
Yuya Nishihara
templater: abort if infinite recursion detected while evaluation (issue4758)...
r27939 def _recursivesymbolblocker(key):
def showrecursion(**args):
raise error.Abort(_("recursive reference '%s' in template") % key)
return showrecursion
Yuya Nishihara
templater: abort if infinite recursion detected while compiling...
r27940 def _runrecursivesymbol(context, mapping, key):
raise error.Abort(_("recursive reference '%s' in template") % key)
Yuya Nishihara
templater: make label() take unknown symbol as color literal...
r28373 def runsymbol(context, mapping, key, default=''):
Yuya Nishihara
templater: look up mapping table through template engine...
r35483 v = context.symbol(mapping, key)
Alexander Plavin
templater: support using templates with non-standard names from map file...
r19770 if v is None:
Yuya Nishihara
templater: abort if infinite recursion detected while evaluation (issue4758)...
r27939 # put poison to cut recursion. we can't move this to parsing phase
# because "x = {x}" is allowed if "x" is a keyword. (issue4758)
safemapping = mapping.copy()
safemapping[key] = _recursivesymbolblocker(key)
Alexander Plavin
templater: support using templates with non-standard names from map file...
r19770 try:
Yuya Nishihara
templater: abort if infinite recursion detected while evaluation (issue4758)...
r27939 v = context.process(key, safemapping)
Alexander Plavin
templater: support using templates with non-standard names from map file...
r19770 except TemplateNotFound:
Yuya Nishihara
templater: make label() take unknown symbol as color literal...
r28373 v = default
Yuya Nishihara
templatekw: add 'requires' flag to switch to exception-safe interface...
r36463 if callable(v) and getattr(v, '_requires', None) is None:
# old templatekw: expand all keywords and resources
Yuya Nishihara
templater: keep default resources per template engine (API)...
r35484 props = context._resources.copy()
props.update(mapping)
Pulkit Goyal
py3: convert dict keys' to str before passing as kwargs...
r35606 return v(**pycompat.strkwargs(props))
Yuya Nishihara
templatekw: add 'requires' flag to switch to exception-safe interface...
r36463 if callable(v):
# new templatekw
try:
return v(context, mapping)
except ResourceUnavailable:
# unsupported keyword is mapped to empty just like unknown keyword
return None
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 return v
Yuya Nishihara
templater: take any string literals as template, but not for rawstring (BC)...
r25596 def buildtemplate(exp, context):
Yuya Nishihara
templater: expand list of parsed templates to template node...
r28547 ctmpl = [compileexp(e, context, methods) for e in exp[1:]]
Yuya Nishihara
templater: take any string literals as template, but not for rawstring (BC)...
r25596 return (runtemplate, ctmpl)
Yuya Nishihara
templater: move runtemplate function out of buildmap/runmap pair...
r25595 def runtemplate(context, mapping, template):
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 for arg in template:
yield evalrawexp(context, mapping, arg)
Yuya Nishihara
templater: move runtemplate function out of buildmap/runmap pair...
r25595
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 def buildfilter(exp, context):
Yuya Nishihara
templater: inline getfilter() to buildfilter()...
r26104 n = getsymbol(exp[2])
if n in context._filters:
filt = context._filters[n]
Yuya Nishihara
templater: add support for keyword arguments...
r31886 arg = compileexp(exp[1], context, methods)
Yuya Nishihara
templater: drop unneeded destructuring of argument tuple at buildfilter...
r26125 return (runfilter, (arg, filt))
Yuya Nishihara
templater: introduce unified filter syntax for unary functions...
r26105 if n in funcs:
f = funcs[n]
Yuya Nishihara
templater: add support for keyword arguments...
r31886 args = _buildfuncargs(exp[1], context, methods, n, f._argspec)
return (f, args)
Yuya Nishihara
templater: inline getfilter() to buildfilter()...
r26104 raise error.ParseError(_("unknown function '%s'") % n)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176
def runfilter(context, mapping, data):
Yuya Nishihara
templater: drop unneeded destructuring of argument tuple at buildfilter...
r26125 arg, filt = data
thing = evalfuncarg(context, mapping, arg)
Neil Kodner
templater: abort when a template filter raises an exception (issue2987)
r17383 try:
Yuya Nishihara
templater: allow piping generator-type function output to filters...
r24280 return filt(thing)
Neil Kodner
templater: abort when a template filter raises an exception (issue2987)
r17383 except (ValueError, AttributeError, TypeError):
Yuya Nishihara
templater: find keyword name more thoroughly on filtering error...
r31927 sym = findsymbolicname(arg)
if sym:
msg = (_("template filter '%s' is not compatible with keyword '%s'")
Augie Fackler
templater: use pycompat.sysbytes to bytes-ify some __name__ attrs...
r34839 % (pycompat.sysbytes(filt.__name__), sym))
Neil Kodner
templater: abort when a template filter raises an exception (issue2987)
r17383 else:
Augie Fackler
templater: use pycompat.sysbytes to bytes-ify some __name__ attrs...
r34839 msg = (_("incompatible use of template filter '%s'")
% pycompat.sysbytes(filt.__name__))
Yuya Nishihara
templater: find keyword name more thoroughly on filtering error...
r31927 raise error.Abort(msg)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176
def buildmap(exp, context):
Yuya Nishihara
templater: do not destructure operands in buildmap()...
r34326 darg = compileexp(exp[1], context, methods)
targ = gettemplate(exp[2], context)
return (runmap, (darg, targ))
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176
def runmap(context, mapping, data):
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 darg, targ = data
d = evalrawexp(context, mapping, darg)
Yuya Nishihara
templater: make _hybrid not callable to avoid conflicting semantics...
r27891 if util.safehasattr(d, 'itermaps'):
Yuya Nishihara
templater: handle exception when applying map operator to non-iterable object...
r28349 diter = d.itermaps()
else:
try:
diter = iter(d)
except TypeError:
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 sym = findsymbolicname(darg)
Yuya Nishihara
templater: use helper function to get name of non-iterable keyword
r34325 if sym:
raise error.ParseError(_("keyword '%s' is not iterable") % sym)
Yuya Nishihara
templater: handle exception when applying map operator to non-iterable object...
r28349 else:
raise error.ParseError(_("%r is not iterable") % d)
Matt Mackall
templating: make new-style templating features work with command line lists
r17631
Yuya Nishihara
templater: provide loop counter as "index" keyword...
r31807 for i, v in enumerate(diter):
Kostia Balytskyi
templater: fix list templating bug...
r28225 lm = mapping.copy()
Yuya Nishihara
templater: provide loop counter as "index" keyword...
r31807 lm['index'] = i
Yuya Nishihara
templater: rename variable "i" to "v" in runmap()...
r31806 if isinstance(v, dict):
lm.update(v)
Weiwen
hgweb: display diff for a changeset against any parents (issue2810)...
r17991 lm['originalnode'] = mapping.get('node')
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 yield evalrawexp(context, lm, targ)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 else:
# v is not an iterable of dicts, this happen when 'key'
# has been fully expanded already and format is useless.
# If so, return the expanded value.
Yuya Nishihara
templater: rename variable "i" to "v" in runmap()...
r31806 yield v
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176
Yuya Nishihara
templater: add dot operator to easily access a sub item...
r34536 def buildmember(exp, context):
darg = compileexp(exp[1], context, methods)
memb = getsymbol(exp[2])
return (runmember, (darg, memb))
def runmember(context, mapping, data):
darg, memb = data
d = evalrawexp(context, mapping, darg)
if util.safehasattr(d, 'tomap'):
lm = mapping.copy()
lm.update(d.tomap())
return runsymbol(context, lm, memb)
Yuya Nishihara
templater: extend dot operator as a short for get(dict, key)
r34537 if util.safehasattr(d, 'get'):
return _getdictitem(d, memb)
Yuya Nishihara
templater: add dot operator to easily access a sub item...
r34536
sym = findsymbolicname(darg)
if sym:
raise error.ParseError(_("keyword '%s' has no member") % sym)
else:
Yuya Nishihara
py3: drop b'' from error message generated by templater.runmember()
r36563 raise error.ParseError(_("%r has no member") % pycompat.bytestr(d))
Yuya Nishihara
templater: add dot operator to easily access a sub item...
r34536
Simon Farnsworth
templater: provide arithmetic operations on integers...
r30115 def buildnegate(exp, context):
arg = compileexp(exp[1], context, exprmethods)
return (runnegate, arg)
def runnegate(context, mapping, data):
data = evalinteger(context, mapping, data,
_('negation needs an integer argument'))
return -data
def buildarithmetic(exp, context, func):
left = compileexp(exp[1], context, exprmethods)
right = compileexp(exp[2], context, exprmethods)
return (runarithmetic, (func, left, right))
def runarithmetic(context, mapping, data):
func, left, right = data
left = evalinteger(context, mapping, left,
_('arithmetic only defined on integers'))
right = evalinteger(context, mapping, right,
_('arithmetic only defined on integers'))
Simon Farnsworth
templater: handle division by zero in arithmetic...
r30116 try:
return func(left, right)
except ZeroDivisionError:
raise error.Abort(_('division by zero is not defined'))
Simon Farnsworth
templater: provide arithmetic operations on integers...
r30115
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 def buildfunc(exp, context):
n = getsymbol(exp[1])
Matt Mackall
templater: use a global funcs table
r14925 if n in funcs:
f = funcs[n]
Yuya Nishihara
templater: add support for keyword arguments...
r31886 args = _buildfuncargs(exp[2], context, exprmethods, n, f._argspec)
Matt Mackall
templater: use a global funcs table
r14925 return (f, args)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 if n in context._filters:
Yuya Nishihara
templater: add support for keyword arguments...
r31886 args = _buildfuncargs(exp[2], context, exprmethods, n, argspec=None)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 if len(args) != 1:
raise error.ParseError(_("filter %s expects one argument") % n)
f = context._filters[n]
Yuya Nishihara
templater: drop unneeded destructuring of argument tuple at buildfilter...
r26125 return (runfilter, (args[0], f))
Sean Farley
templater: raise error for unknown func...
r20857 raise error.ParseError(_("unknown function '%s'") % n)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176
Yuya Nishihara
templater: add support for keyword arguments...
r31886 def _buildfuncargs(exp, context, curmethods, funcname, argspec):
"""Compile parsed tree of function arguments into list or dict of
Yuya Nishihara
parser: extend buildargsdict() to support arbitrary number of **kwargs...
r31921 (func, data) pairs
>>> context = engine(lambda t: (runsymbol, t))
>>> def fargs(expr, argspec):
... x = _parseexpr(expr)
... n = getsymbol(x[1])
... return _buildfuncargs(x[2], context, exprmethods, n, argspec)
Yuya Nishihara
doctest: coerce dict.keys() to list...
r34141 >>> list(fargs(b'a(l=1, k=2)', b'k l m').keys())
Yuya Nishihara
parser: preserve order of keyword arguments...
r31922 ['l', 'k']
Yuya Nishihara
doctest: bulk-replace string literals with b'' for Python 3...
r34133 >>> args = fargs(b'a(opts=1, k=2)', b'**opts')
Yuya Nishihara
doctest: coerce dict.keys() to list...
r34141 >>> list(args.keys()), list(args[b'opts'].keys())
Yuya Nishihara
parser: preserve order of keyword arguments...
r31922 (['opts'], ['opts', 'k'])
Yuya Nishihara
parser: extend buildargsdict() to support arbitrary number of **kwargs...
r31921 """
Yuya Nishihara
templater: add support for keyword arguments...
r31886 def compiledict(xs):
Yuya Nishihara
parser: preserve order of keyword arguments...
r31922 return util.sortdict((k, compileexp(x, context, curmethods))
for k, x in xs.iteritems())
Yuya Nishihara
templater: add support for keyword arguments...
r31886 def compilelist(xs):
return [compileexp(x, context, curmethods) for x in xs]
if not argspec:
# filter or function with no argspec: return list of positional args
return compilelist(getlist(exp))
# function with argspec: return dict of named args
Yuya Nishihara
parser: extend buildargsdict() to support arbitrary number of **kwargs...
r31921 _poskeys, varkey, _keys, optkey = argspec = parser.splitargspec(argspec)
Yuya Nishihara
templater: add support for keyword arguments...
r31886 treeargs = parser.buildargsdict(getlist(exp), funcname, argspec,
keyvaluenode='keyvalue', keynode='symbol')
Yuya Nishihara
parser: preserve order of keyword arguments...
r31922 compargs = util.sortdict()
Yuya Nishihara
templater: add support for keyword arguments...
r31886 if varkey:
compargs[varkey] = compilelist(treeargs.pop(varkey))
Yuya Nishihara
parser: extend buildargsdict() to support arbitrary number of **kwargs...
r31921 if optkey:
compargs[optkey] = compiledict(treeargs.pop(optkey))
Yuya Nishihara
templater: add support for keyword arguments...
r31886 compargs.update(compiledict(treeargs))
return compargs
Yuya Nishihara
templater: add parsing rule for key-value pair...
r31885 def buildkeyvaluepair(exp, content):
raise error.ParseError(_("can't use a key-value pair in this context"))
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 # dict of template built-in functions
funcs = {}
templatefunc = registrar.templatefunc(funcs)
@templatefunc('date(date[, fmt])')
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 def date(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Format a date. See :hg:`help dates` for formatting
Yuya Nishihara
templatefilters: remove redundant 'date' and 'strip' filters...
r26106 strings. The default is a Unix date format, including the timezone:
"Mon Sep 04 15:13:13 2006 0700"."""
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 if not (1 <= len(args) <= 2):
FUJIWARA Katsunori
i18n: add i18n comment to error messages of template functions
r23112 # i18n: "date" is a keyword
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 raise error.ParseError(_("date expects one or two arguments"))
Yuya Nishihara
templater: make date() use helper function to evaluate argument...
r28334 date = evalfuncarg(context, mapping, args[0])
Yuya Nishihara
templater: fix crash by passing invalid object to date() function...
r24903 fmt = None
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 if len(args) == 2:
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 fmt = evalstring(context, mapping, args[1])
Yuya Nishihara
templater: fix crash by passing invalid object to date() function...
r24903 try:
if fmt is None:
Boris Feld
util: extract all date-related utils in utils/dateutil module...
r36625 return dateutil.datestr(date)
Yuya Nishihara
templater: fix crash by passing invalid object to date() function...
r24903 else:
Boris Feld
util: extract all date-related utils in utils/dateutil module...
r36625 return dateutil.datestr(date, fmt)
Yuya Nishihara
templater: fix crash by passing invalid object to date() function...
r24903 except (TypeError, ValueError):
# i18n: "date" is a keyword
raise error.ParseError(_("date expects a date information"))
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390
Yuya Nishihara
templater: add shorthand for building a dict like {"key": key}...
r31928 @templatefunc('dict([[key=]value...])', argspec='*args **kwargs')
Yuya Nishihara
templater: add dict() constructor...
r31926 def dict_(context, mapping, args):
Yuya Nishihara
templater: add shorthand for building a dict like {"key": key}...
r31928 """Construct a dict from key-value pairs. A key may be omitted if
a value expression can provide an unambiguous name."""
Yuya Nishihara
templater: add dict() constructor...
r31926 data = util.sortdict()
Yuya Nishihara
templater: add shorthand for building a dict like {"key": key}...
r31928
for v in args['args']:
k = findsymbolicname(v)
if not k:
raise error.ParseError(_('dict key cannot be inferred'))
if k in data or k in args['kwargs']:
raise error.ParseError(_("duplicated dict key '%s' inferred") % k)
data[k] = evalfuncarg(context, mapping, v)
Yuya Nishihara
templater: add dict() constructor...
r31926 data.update((k, evalfuncarg(context, mapping, v))
for k, v in args['kwargs'].iteritems())
return templatekw.hybriddict(data)
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('diff([includepattern [, excludepattern]])')
FUJIWARA Katsunori
templater: add "diff" template function...
r22434 def diff(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Show a diff, optionally
Gregory Szorc
templater: add consistent docstrings to functions...
r24586 specifying files to include or exclude."""
FUJIWARA Katsunori
templater: add "diff" template function...
r22434 if len(args) > 2:
# i18n: "diff" is a keyword
timeless
grammar: favor zero, one, two over ... or no
r27293 raise error.ParseError(_("diff expects zero, one, or two arguments"))
FUJIWARA Katsunori
templater: add "diff" template function...
r22434
def getpatterns(i):
if i < len(args):
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 s = evalstring(context, mapping, args[i]).strip()
FUJIWARA Katsunori
templater: add "diff" template function...
r22434 if s:
return [s]
return []
Yuya Nishihara
templater: look up mapping table through template engine...
r35483 ctx = context.resource(mapping, 'ctx')
FUJIWARA Katsunori
templater: add "diff" template function...
r22434 chunks = ctx.diff(match=ctx.match([], getpatterns(0), getpatterns(1)))
return ''.join(chunks)
Yuya Nishihara
templater: add experimental support for extdata...
r34459 @templatefunc('extdata(source)', argspec='source')
def extdata(context, mapping, args):
"""Show a text read from the specified extdata source. (EXPERIMENTAL)"""
if 'source' not in args:
# i18n: "extdata" is a keyword
raise error.ParseError(_('extdata expects one argument'))
source = evalstring(context, mapping, args['source'])
Yuya Nishihara
templater: look up mapping table through template engine...
r35483 cache = context.resource(mapping, 'cache').setdefault('extdata', {})
ctx = context.resource(mapping, 'ctx')
Yuya Nishihara
templater: add experimental support for extdata...
r34459 if source in cache:
data = cache[source]
else:
data = cache[source] = scmutil.extdatasource(ctx.repo(), source)
return data.get(ctx.rev(), '')
Hannes Oldenburg
templates: add built-in files() function...
r30008 @templatefunc('files(pattern)')
def files(context, mapping, args):
"""All files of the current changeset matching the pattern. See
:hg:`help patterns`."""
if not len(args) == 1:
# i18n: "files" is a keyword
raise error.ParseError(_("files expects one argument"))
raw = evalstring(context, mapping, args[0])
Yuya Nishihara
templater: look up mapping table through template engine...
r35483 ctx = context.resource(mapping, 'ctx')
Hannes Oldenburg
templates: add built-in files() function...
r30008 m = ctx.match([raw])
files = list(ctx.matches(m))
Yuya Nishihara
templatekw: switch most of showlist template keywords to new API (issue5779)...
r36609 return templatekw.compatlist(context, mapping, "file", files)
Hannes Oldenburg
templates: add built-in files() function...
r30008
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('fill(text[, width[, initialident[, hangindent]]])')
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 def fill(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Fill many
Gregory Szorc
templater: add consistent docstrings to functions...
r24586 paragraphs with optional indentation. See the "fill" filter."""
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 if not (1 <= len(args) <= 4):
FUJIWARA Katsunori
i18n: add i18n comment to error messages of template functions
r23112 # i18n: "fill" is a keyword
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 raise error.ParseError(_("fill expects one to four arguments"))
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 text = evalstring(context, mapping, args[0])
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 width = 76
initindent = ''
hangindent = ''
if 2 <= len(args) <= 4:
Yuya Nishihara
templater: factor out function that evaluates argument as integer...
r28343 width = evalinteger(context, mapping, args[1],
# i18n: "fill" is a keyword
_("fill expects an integer width"))
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 try:
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 initindent = evalstring(context, mapping, args[2])
hangindent = evalstring(context, mapping, args[3])
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 except IndexError:
pass
return templatefilters.fill(text, width, initindent, hangindent)
Yuya Nishihara
templater: port formatnode filter from changeset_templater...
r31169 @templatefunc('formatnode(node)')
def formatnode(context, mapping, args):
"""Obtain the preferred form of a changeset hash. (DEPRECATED)"""
if len(args) != 1:
# i18n: "formatnode" is a keyword
raise error.ParseError(_("formatnode expects one argument"))
Yuya Nishihara
templater: look up mapping table through template engine...
r35483 ui = context.resource(mapping, 'ui')
Yuya Nishihara
templater: port formatnode filter from changeset_templater...
r31169 node = evalstring(context, mapping, args[0])
if ui.debugflag:
return node
return templatefilters.short(node)
Yuya Nishihara
templater: port pad() to take keyword arguments...
r31887 @templatefunc('pad(text, width[, fillchar=\' \'[, left=False]])',
argspec='text width fillchar left')
Durham Goode
template: add pad function for padding output...
r20370 def pad(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Pad text with a
Gregory Szorc
templater: add consistent docstrings to functions...
r24586 fill character."""
Yuya Nishihara
templater: port pad() to take keyword arguments...
r31887 if 'text' not in args or 'width' not in args:
FUJIWARA Katsunori
i18n: add i18n comment to error messages of template functions
r23112 # i18n: "pad" is a keyword
Durham Goode
template: add pad function for padding output...
r20370 raise error.ParseError(_("pad() expects two to four arguments"))
Yuya Nishihara
templater: port pad() to take keyword arguments...
r31887 width = evalinteger(context, mapping, args['width'],
Yuya Nishihara
templater: fix pad() to evaluate int argument and handle error
r28345 # i18n: "pad" is a keyword
_("pad() expects an integer width"))
Durham Goode
template: add pad function for padding output...
r20370
Yuya Nishihara
templater: port pad() to take keyword arguments...
r31887 text = evalstring(context, mapping, args['text'])
Durham Goode
template: add pad function for padding output...
r20370
Yuya Nishihara
templater: rename "right" argument of pad() function...
r29818 left = False
Durham Goode
template: add pad function for padding output...
r20370 fillchar = ' '
Yuya Nishihara
templater: port pad() to take keyword arguments...
r31887 if 'fillchar' in args:
fillchar = evalstring(context, mapping, args['fillchar'])
Yuya Nishihara
templater: make pad() strip color codes before computing width (issue5416)...
r31521 if len(color.stripeffects(fillchar)) != 1:
Yuya Nishihara
templater: reject bad fillchar argument passed to pad()...
r31519 # i18n: "pad" is a keyword
raise error.ParseError(_("pad() expects a single fill character"))
Yuya Nishihara
templater: port pad() to take keyword arguments...
r31887 if 'left' in args:
left = evalboolean(context, mapping, args['left'])
Durham Goode
template: add pad function for padding output...
r20370
Yuya Nishihara
templater: make pad() strip color codes before computing width (issue5416)...
r31521 fillwidth = width - encoding.colwidth(color.stripeffects(text))
Yuya Nishihara
templater: make pad() compute actual width...
r31520 if fillwidth <= 0:
return text
Yuya Nishihara
templater: rename "right" argument of pad() function...
r29818 if left:
Yuya Nishihara
templater: make pad() compute actual width...
r31520 return fillchar * fillwidth + text
Durham Goode
template: add pad function for padding output...
r20370 else:
Yuya Nishihara
templater: make pad() compute actual width...
r31520 return text + fillchar * fillwidth
Durham Goode
template: add pad function for padding output...
r20370
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('indent(text, indentchars[, firstline])')
Ryan McElroy
templater: introduce indent function
r25489 def indent(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Indents all non-empty lines
Ryan McElroy
templater: introduce indent function
r25489 with the characters given in the indentchars string. An optional
third parameter will override the indent for the first line only
if present."""
if not (2 <= len(args) <= 3):
# i18n: "indent" is a keyword
raise error.ParseError(_("indent() expects two or three arguments"))
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 text = evalstring(context, mapping, args[0])
indent = evalstring(context, mapping, args[1])
Ryan McElroy
templater: introduce indent function
r25489
if len(args) == 3:
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 firstline = evalstring(context, mapping, args[2])
Ryan McElroy
templater: introduce indent function
r25489 else:
firstline = indent
# the indent function doesn't indent the first line, so we do it here
return templatefilters.indent(firstline + text, indent)
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('get(dict, key)')
Benoit Boissinot
templater: add get() function to access dict element (e.g. extra)
r18582 def get(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Get an attribute/key from an object. Some keywords
Gregory Szorc
templater: add consistent docstrings to functions...
r24586 are complex types. This function allows you to obtain the value of an
timeless@mozdev.org
templater: fix get English
r26197 attribute on these types."""
Benoit Boissinot
templater: add get() function to access dict element (e.g. extra)
r18582 if len(args) != 2:
# i18n: "get" is a keyword
raise error.ParseError(_("get() expects two arguments"))
Yuya Nishihara
templater: fix get() to evaluate arguments eagerly...
r28331 dictarg = evalfuncarg(context, mapping, args[0])
Benoit Boissinot
templater: add get() function to access dict element (e.g. extra)
r18582 if not util.safehasattr(dictarg, 'get'):
# i18n: "get" is a keyword
raise error.ParseError(_("get() expects a dict as first argument"))
Yuya Nishihara
templater: fix get() to evaluate arguments eagerly...
r28331 key = evalfuncarg(context, mapping, args[1])
Yuya Nishihara
templater: extend dot operator as a short for get(dict, key)
r34537 return _getdictitem(dictarg, key)
def _getdictitem(dictarg, key):
Yuya Nishihara
templater: wrap get/min/max result so map operation can apply to element...
r34535 val = dictarg.get(key)
if val is None:
return
return templatekw.wraphybridvalue(dictarg, key, val)
Benoit Boissinot
templater: add get() function to access dict element (e.g. extra)
r18582
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('if(expr, then[, else])')
Matt Mackall
templater: add if/ifeq conditionals
r17636 def if_(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Conditionally execute based on the result of
Gregory Szorc
templater: add consistent docstrings to functions...
r24586 an expression."""
Matt Mackall
templater: add if/ifeq conditionals
r17636 if not (2 <= len(args) <= 3):
FUJIWARA Katsunori
i18n: add "i18n" comment to error messages of template functions
r17890 # i18n: "if" is a keyword
Matt Mackall
templater: add if/ifeq conditionals
r17636 raise error.ParseError(_("if expects two or three arguments"))
Yuya Nishihara
templater: fix if() to not evaluate False as bool('False')...
r29816 test = evalboolean(context, mapping, args[0])
Matt Mackall
templater: add if/ifeq conditionals
r17636 if test:
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 yield evalrawexp(context, mapping, args[1])
Matt Mackall
templater: add if/ifeq conditionals
r17636 elif len(args) == 3:
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 yield evalrawexp(context, mapping, args[2])
Matt Mackall
templater: add if/ifeq conditionals
r17636
av6
templater: use "needle" and "haystack" as (meta-)variables for ifcontains()...
r30049 @templatefunc('ifcontains(needle, haystack, then[, else])')
Durham Goode
template: add ifcontains template function...
r20518 def ifcontains(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Conditionally execute based
av6
templater: use "needle" and "haystack" as (meta-)variables for ifcontains()...
r30049 on whether the item "needle" is in "haystack"."""
Durham Goode
template: add ifcontains template function...
r20518 if not (3 <= len(args) <= 4):
# i18n: "ifcontains" is a keyword
raise error.ParseError(_("ifcontains expects three or four arguments"))
av6
templater: use "needle" and "haystack" as (meta-)variables for ifcontains()...
r30049 haystack = evalfuncarg(context, mapping, args[1])
Yuya Nishihara
templater: fix ifcontains() to handle type mismatch gracefully...
r34660 try:
needle = evalastype(context, mapping, args[0],
getattr(haystack, 'keytype', None) or bytes)
found = (needle in haystack)
except error.ParseError:
found = False
Durham Goode
template: add ifcontains template function...
r20518
Yuya Nishihara
templater: fix ifcontains() to handle type mismatch gracefully...
r34660 if found:
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 yield evalrawexp(context, mapping, args[2])
Durham Goode
template: add ifcontains template function...
r20518 elif len(args) == 4:
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 yield evalrawexp(context, mapping, args[3])
Durham Goode
template: add ifcontains template function...
r20518
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('ifeq(expr1, expr2, then[, else])')
Matt Mackall
templater: add if/ifeq conditionals
r17636 def ifeq(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Conditionally execute based on
Gregory Szorc
templater: add consistent docstrings to functions...
r24586 whether 2 items are equivalent."""
Matt Mackall
templater: add if/ifeq conditionals
r17636 if not (3 <= len(args) <= 4):
FUJIWARA Katsunori
i18n: add "i18n" comment to error messages of template functions
r17890 # i18n: "ifeq" is a keyword
Matt Mackall
templater: add if/ifeq conditionals
r17636 raise error.ParseError(_("ifeq expects three or four arguments"))
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 test = evalstring(context, mapping, args[0])
match = evalstring(context, mapping, args[1])
Matt Mackall
templater: add if/ifeq conditionals
r17636 if test == match:
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 yield evalrawexp(context, mapping, args[2])
Matt Mackall
templater: add if/ifeq conditionals
r17636 elif len(args) == 4:
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 yield evalrawexp(context, mapping, args[3])
Matt Mackall
templater: add if/ifeq conditionals
r17636
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('join(list, sep)')
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 def join(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Join items in a list with a delimiter."""
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 if not (1 <= len(args) <= 2):
# i18n: "join" is a keyword
raise error.ParseError(_("join expects one or two arguments"))
Yuya Nishihara
templater: extract helper to just evaluate template expression...
r34327 # TODO: perhaps this should be evalfuncarg(), but it can't because hgweb
# abuses generator as a keyword that returns a list of dicts.
joinset = evalrawexp(context, mapping, args[0])
Yuya Nishihara
templatekw: add new-style template expansion to {manifest}...
r34331 joinset = templatekw.unwrapvalue(joinset)
Yuya Nishihara
templatekw: just pass underlying value (or key) to joinfmt() function...
r34329 joinfmt = getattr(joinset, 'joinfmt', pycompat.identity)
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 joiner = " "
if len(args) > 1:
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 joiner = evalstring(context, mapping, args[1])
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390
first = True
Yuya Nishihara
py3: fix join(), min(), and max() template functions over string...
r36562 for x in pycompat.maybebytestr(joinset):
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 if first:
first = False
else:
yield joiner
Yuya Nishihara
templatekw: just pass underlying value (or key) to joinfmt() function...
r34329 yield joinfmt(x)
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('label(label, expr)')
Sean Farley
templater: add no-op template function 'label'
r18289 def label(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Apply a label to generated content. Content with
Gregory Szorc
templater: add consistent docstrings to functions...
r24586 a label applied can result in additional post-processing, such as
automatic colorization."""
Sean Farley
templater: add no-op template function 'label'
r18289 if len(args) != 2:
# i18n: "label" is a keyword
raise error.ParseError(_("label expects two arguments"))
Yuya Nishihara
templater: look up mapping table through template engine...
r35483 ui = context.resource(mapping, 'ui')
Yuya Nishihara
templater: move label() function from color extension...
r28374 thing = evalstring(context, mapping, args[1])
# preserve unknown symbol as literal so effects like 'red', 'bold',
# etc. don't need to be quoted
label = evalstringliteral(context, mapping, args[0])
Kostia Balytskyi
formatter: make labels work with templated output...
r28384 return ui.label(thing, label)
Sean Farley
templater: add no-op template function 'label'
r18289
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('latesttag([pattern])')
Matt Harbison
templater: introduce {latesttag()} function to match a pattern (issue4184)...
r26485 def latesttag(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """The global tags matching the given pattern on the
Matt Harbison
templatekw: clarify the result of {latesttag} when no tag exists...
r31850 most recent globally tagged ancestor of this changeset.
If no such tags exist, the "{tag}" template resolves to
the string "null"."""
Matt Harbison
templater: introduce {latesttag()} function to match a pattern (issue4184)...
r26485 if len(args) > 1:
# i18n: "latesttag" is a keyword
raise error.ParseError(_("latesttag expects at most one argument"))
pattern = None
if len(args) == 1:
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 pattern = evalstring(context, mapping, args[0])
Yuya Nishihara
templatekw: switch latesttags template keywords to new API
r36614 return templatekw.showlatesttags(context, mapping, pattern)
Matt Harbison
templater: introduce {latesttag()} function to match a pattern (issue4184)...
r26485
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('localdate(date[, tz])')
Yuya Nishihara
templater: port localdate filter to a function...
r26127 def localdate(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Converts a date to the specified timezone.
Yuya Nishihara
templater: add optional timezone argument to localdate()...
r26128 The default is local date."""
if not (1 <= len(args) <= 2):
Yuya Nishihara
templater: port localdate filter to a function...
r26127 # i18n: "localdate" is a keyword
Yuya Nishihara
templater: add optional timezone argument to localdate()...
r26128 raise error.ParseError(_("localdate expects one or two arguments"))
Yuya Nishihara
templater: port localdate filter to a function...
r26127
date = evalfuncarg(context, mapping, args[0])
try:
Boris Feld
util: extract all date-related utils in utils/dateutil module...
r36625 date = dateutil.parsedate(date)
Yuya Nishihara
templater: port localdate filter to a function...
r26127 except AttributeError: # not str nor date tuple
# i18n: "localdate" is a keyword
raise error.ParseError(_("localdate expects a date information"))
Yuya Nishihara
templater: add optional timezone argument to localdate()...
r26128 if len(args) >= 2:
tzoffset = None
tz = evalfuncarg(context, mapping, args[1])
Yuya Nishihara
py3: replace type 'str' by 'bytes' in templater.py
r36568 if isinstance(tz, bytes):
Boris Feld
util: extract all date-related utils in utils/dateutil module...
r36625 tzoffset, remainder = dateutil.parsetimezone(tz)
Matt Mackall
date: refactor timezone parsing...
r29636 if remainder:
tzoffset = None
Yuya Nishihara
templater: add optional timezone argument to localdate()...
r26128 if tzoffset is None:
try:
tzoffset = int(tz)
except (TypeError, ValueError):
# i18n: "localdate" is a keyword
raise error.ParseError(_("localdate expects a timezone"))
else:
Boris Feld
util: extract all date-related utils in utils/dateutil module...
r36625 tzoffset = dateutil.makedate()[1]
Yuya Nishihara
templater: port localdate filter to a function...
r26127 return (date[0], tzoffset)
Boris Feld
template: compute dates in obsfatedate...
r33997 @templatefunc('max(iterable)')
def max_(context, mapping, args, **kwargs):
"""Return the max of an iterable"""
if len(args) != 1:
# i18n: "max" is a keyword
av6
templater: fix "one arguments"
r35387 raise error.ParseError(_("max expects one argument"))
Boris Feld
template: compute dates in obsfatedate...
r33997
iterable = evalfuncarg(context, mapping, args[0])
try:
Yuya Nishihara
py3: fix join(), min(), and max() template functions over string...
r36562 x = max(pycompat.maybebytestr(iterable))
Boris Feld
template: compute dates in obsfatedate...
r33997 except (TypeError, ValueError):
# i18n: "max" is a keyword
raise error.ParseError(_("max first argument should be an iterable"))
Yuya Nishihara
templater: wrap get/min/max result so map operation can apply to element...
r34535 return templatekw.wraphybridvalue(iterable, x, x)
Boris Feld
template: compute dates in obsfatedate...
r33997
@templatefunc('min(iterable)')
def min_(context, mapping, args, **kwargs):
"""Return the min of an iterable"""
if len(args) != 1:
# i18n: "min" is a keyword
av6
templater: fix "one arguments"
r35387 raise error.ParseError(_("min expects one argument"))
Boris Feld
template: compute dates in obsfatedate...
r33997
iterable = evalfuncarg(context, mapping, args[0])
try:
Yuya Nishihara
py3: fix join(), min(), and max() template functions over string...
r36562 x = min(pycompat.maybebytestr(iterable))
Boris Feld
template: compute dates in obsfatedate...
r33997 except (TypeError, ValueError):
# i18n: "min" is a keyword
raise error.ParseError(_("min first argument should be an iterable"))
Yuya Nishihara
templater: wrap get/min/max result so map operation can apply to element...
r34535 return templatekw.wraphybridvalue(iterable, x, x)
Boris Feld
template: compute dates in obsfatedate...
r33997
Simon Farnsworth
templater: provide arithmetic operations on integers...
r30115 @templatefunc('mod(a, b)')
def mod(context, mapping, args):
"""Calculate a mod b such that a / b + a mod b == a"""
if not len(args) == 2:
# i18n: "mod" is a keyword
raise error.ParseError(_("mod expects two arguments"))
Simon Farnsworth
templater: handle division by zero in arithmetic...
r30116 func = lambda a, b: a % b
return runarithmetic(context, mapping, (func, args[0], args[1]))
Simon Farnsworth
templater: provide arithmetic operations on integers...
r30115
Martin von Zweigbergk
templates: introduce a obsfateoperation() function...
r34288 @templatefunc('obsfateoperations(markers)')
def obsfateoperations(context, mapping, args):
"""Compute obsfate related information based on markers (EXPERIMENTAL)"""
if len(args) != 1:
# i18n: "obsfateoperations" is a keyword
av6
templater: fix "one arguments"
r35387 raise error.ParseError(_("obsfateoperations expects one argument"))
Martin von Zweigbergk
templates: introduce a obsfateoperation() function...
r34288
markers = evalfuncarg(context, mapping, args[0])
try:
data = obsutil.markersoperations(markers)
return templatekw.hybridlist(data, name='operation')
except (TypeError, KeyError):
# i18n: "obsfateoperations" is a keyword
errmsg = _("obsfateoperations first argument should be an iterable")
raise error.ParseError(errmsg)
Boris Feld
template: compute dates in obsfatedate...
r33997 @templatefunc('obsfatedate(markers)')
def obsfatedate(context, mapping, args):
"""Compute obsfate related information based on markers (EXPERIMENTAL)"""
if len(args) != 1:
# i18n: "obsfatedate" is a keyword
av6
templater: fix "one arguments"
r35387 raise error.ParseError(_("obsfatedate expects one argument"))
Boris Feld
template: compute dates in obsfatedate...
r33997
markers = evalfuncarg(context, mapping, args[0])
try:
data = obsutil.markersdates(markers)
return templatekw.hybridlist(data, name='date', fmt='%d %d')
except (TypeError, KeyError):
# i18n: "obsfatedate" is a keyword
errmsg = _("obsfatedate first argument should be an iterable")
raise error.ParseError(errmsg)
Boris Feld
template: compute user in obsfateusers...
r33996 @templatefunc('obsfateusers(markers)')
def obsfateusers(context, mapping, args):
"""Compute obsfate related information based on markers (EXPERIMENTAL)"""
if len(args) != 1:
# i18n: "obsfateusers" is a keyword
av6
templater: fix "one arguments"
r35387 raise error.ParseError(_("obsfateusers expects one argument"))
Boris Feld
template: compute user in obsfateusers...
r33996
markers = evalfuncarg(context, mapping, args[0])
try:
data = obsutil.markersusers(markers)
return templatekw.hybridlist(data, name='user')
except (TypeError, KeyError, ValueError):
# i18n: "obsfateusers" is a keyword
msg = _("obsfateusers first argument should be an iterable of "
"obsmakers")
raise error.ParseError(msg)
Boris Feld
obsfate: makes successorsetverb takes the markers as argument...
r35011 @templatefunc('obsfateverb(successors, markers)')
Boris Feld
template: compute verb in obsfateverb...
r33995 def obsfateverb(context, mapping, args):
"""Compute obsfate related information based on successors (EXPERIMENTAL)"""
Boris Feld
obsfate: makes successorsetverb takes the markers as argument...
r35011 if len(args) != 2:
Boris Feld
template: compute verb in obsfateverb...
r33995 # i18n: "obsfateverb" is a keyword
Boris Feld
obsfate: makes successorsetverb takes the markers as argument...
r35011 raise error.ParseError(_("obsfateverb expects two arguments"))
Boris Feld
template: compute verb in obsfateverb...
r33995
successors = evalfuncarg(context, mapping, args[0])
Boris Feld
obsfate: makes successorsetverb takes the markers as argument...
r35011 markers = evalfuncarg(context, mapping, args[1])
Boris Feld
template: compute verb in obsfateverb...
r33995
try:
Boris Feld
obsfate: makes successorsetverb takes the markers as argument...
r35011 return obsutil.obsfateverb(successors, markers)
Boris Feld
template: compute verb in obsfateverb...
r33995 except TypeError:
# i18n: "obsfateverb" is a keyword
errmsg = _("obsfateverb first argument should be countable")
raise error.ParseError(errmsg)
Yuya Nishihara
templater: add relpath() to convert repo path to relative path (issue5394)...
r30083 @templatefunc('relpath(path)')
def relpath(context, mapping, args):
"""Convert a repository-absolute path into a filesystem path relative to
the current working directory."""
if len(args) != 1:
# i18n: "relpath" is a keyword
raise error.ParseError(_("relpath expects one argument"))
Yuya Nishihara
templater: look up mapping table through template engine...
r35483 repo = context.resource(mapping, 'ctx').repo()
Yuya Nishihara
templater: add relpath() to convert repo path to relative path (issue5394)...
r30083 path = evalstring(context, mapping, args[0])
return repo.pathto(path)
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('revset(query[, formatargs...])')
Durham Goode
template: add revset() template function...
r20519 def revset(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Execute a revision set query. See
Gregory Szorc
templater: add consistent docstrings to functions...
r24586 :hg:`help revset`."""
Durham Goode
template: add revset() template function...
r20519 if not len(args) > 0:
# i18n: "revset" is a keyword
raise error.ParseError(_("revset expects one or more arguments"))
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 raw = evalstring(context, mapping, args[0])
Yuya Nishihara
templater: look up mapping table through template engine...
r35483 ctx = context.resource(mapping, 'ctx')
Matt Harbison
templater: replace 'ctx._repo' with 'ctx.repo()'
r24337 repo = ctx.repo()
Durham Goode
template: add revset() template function...
r20519
FUJIWARA Katsunori
templater: enable alias predicates to be used in "revset()" function...
r22304 def query(expr):
Gregory Szorc
revset: pass repo when passing ui...
r33554 m = revsetmod.match(repo.ui, expr, repo=repo)
Yuya Nishihara
revset: make match function initiate query from full set by default...
r24114 return m(repo)
FUJIWARA Katsunori
templater: enable alias predicates to be used in "revset()" function...
r22304
Durham Goode
template: add revset() template function...
r20519 if len(args) > 1:
Yuya Nishihara
templater: fix revset() to evaluate format arguments eagerly...
r28333 formatargs = [evalfuncarg(context, mapping, a) for a in args[1:]]
Yuya Nishihara
revset: split language services to revsetlang module (API)...
r31024 revs = query(revsetlang.formatspec(raw, *formatargs))
Yuya Nishihara
templater: factor out type conversion of revset() result...
r28178 revs = list(revs)
Durham Goode
template: add revset() template function...
r20519 else:
Yuya Nishihara
templater: look up mapping table through template engine...
r35483 cache = context.resource(mapping, 'cache')
revsetcache = cache.setdefault("revsetcache", {})
Durham Goode
template: add revset() template function...
r20519 if raw in revsetcache:
revs = revsetcache[raw]
else:
FUJIWARA Katsunori
templater: enable alias predicates to be used in "revset()" function...
r22304 revs = query(raw)
Yuya Nishihara
templater: factor out type conversion of revset() result...
r28178 revs = list(revs)
Durham Goode
template: add revset() template function...
r20519 revsetcache[raw] = revs
Yuya Nishihara
templatekw: switch revset() to new API
r36613 return templatekw.showrevslist(context, mapping, "revision", revs)
Durham Goode
template: add revset() template function...
r20519
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('rstdoc(text, style)')
Dan Villiom Podlaski Christiansen
hgweb: generate HTML documentation...
r18747 def rstdoc(context, mapping, args):
Mads Kiilerich
spelling: fixes of non-dictionary words
r30332 """Format reStructuredText."""
Dan Villiom Podlaski Christiansen
hgweb: generate HTML documentation...
r18747 if len(args) != 2:
# i18n: "rstdoc" is a keyword
raise error.ParseError(_("rstdoc expects two arguments"))
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 text = evalstring(context, mapping, args[0])
style = evalstring(context, mapping, args[1])
Dan Villiom Podlaski Christiansen
hgweb: generate HTML documentation...
r18747
Alexander Plavin
hgweb: make help verbose again (issue3899)...
r19079 return minirst.format(text, style=style, keep=['verbose'])
Dan Villiom Podlaski Christiansen
hgweb: generate HTML documentation...
r18747
Yuya Nishihara
templater: add support for keyword arguments...
r31886 @templatefunc('separate(sep, args)', argspec='sep *args')
Martin von Zweigbergk
templater: add separate() template function...
r29085 def separate(context, mapping, args):
"""Add a separator between non-empty arguments."""
Yuya Nishihara
templater: add support for keyword arguments...
r31886 if 'sep' not in args:
Martin von Zweigbergk
templater: add separate() template function...
r29085 # i18n: "separate" is a keyword
raise error.ParseError(_("separate expects at least one argument"))
Yuya Nishihara
templater: add support for keyword arguments...
r31886 sep = evalstring(context, mapping, args['sep'])
Martin von Zweigbergk
templater: add separate() template function...
r29085 first = True
Yuya Nishihara
templater: add support for keyword arguments...
r31886 for arg in args['args']:
Martin von Zweigbergk
templater: add separate() template function...
r29085 argstr = evalstring(context, mapping, arg)
if not argstr:
continue
if first:
first = False
else:
yield sep
yield argstr
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('shortest(node, minlength=4)')
Durham Goode
template: add shortest(node) template function...
r20369 def shortest(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Obtain the shortest representation of
Gregory Szorc
templater: add consistent docstrings to functions...
r24586 a node."""
Durham Goode
template: add shortest(node) template function...
r20369 if not (1 <= len(args) <= 2):
FUJIWARA Katsunori
i18n: add i18n comment to error messages of template functions
r23112 # i18n: "shortest" is a keyword
Durham Goode
template: add shortest(node) template function...
r20369 raise error.ParseError(_("shortest() expects one or two arguments"))
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 node = evalstring(context, mapping, args[0])
Durham Goode
template: add shortest(node) template function...
r20369
minlength = 4
if len(args) > 1:
Yuya Nishihara
templater: fix shortest() to evaluate int argument and handle error
r28346 minlength = evalinteger(context, mapping, args[1],
# i18n: "shortest" is a keyword
_("shortest() expects an integer minlength"))
Durham Goode
template: add shortest(node) template function...
r20369
Yuya Nishihara
templater: use unfiltered changelog to calculate shortest() at constant time...
r30232 # _partialmatch() of filtered changelog could take O(len(repo)) time,
# which would be unacceptably slow. so we look for hash collision in
# unfiltered space, which means some hashes may be slightly longer.
Yuya Nishihara
templater: look up mapping table through template engine...
r35483 cl = context.resource(mapping, 'ctx')._repo.unfiltered().changelog
Martin von Zweigbergk
templater: extract shortest() logic from template function...
r34247 return cl.shortest(node, minlength)
Durham Goode
template: add shortest(node) template function...
r20369
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('strip(text[, chars])')
Alexander Plavin
templater: add strip function with chars as an extra argument...
r19330 def strip(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Strip characters from a string. By default,
Yuya Nishihara
templatefilters: remove redundant 'date' and 'strip' filters...
r26106 strips all leading and trailing whitespace."""
Alexander Plavin
templater: add strip function with chars as an extra argument...
r19330 if not (1 <= len(args) <= 2):
FUJIWARA Katsunori
i18n: add i18n comment to error messages of template functions
r23112 # i18n: "strip" is a keyword
Alexander Plavin
templater: add strip function with chars as an extra argument...
r19330 raise error.ParseError(_("strip expects one or two arguments"))
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 text = evalstring(context, mapping, args[0])
Alexander Plavin
templater: add strip function with chars as an extra argument...
r19330 if len(args) == 2:
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 chars = evalstring(context, mapping, args[1])
Alexander Plavin
templater: add strip function with chars as an extra argument...
r19330 return text.strip(chars)
return text.strip()
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('sub(pattern, replacement, expression)')
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 def sub(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Perform text substitution
Gregory Szorc
templater: add consistent docstrings to functions...
r24586 using regular expressions."""
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390 if len(args) != 3:
# i18n: "sub" is a keyword
raise error.ParseError(_("sub expects three arguments"))
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 pat = evalstring(context, mapping, args[0])
rpl = evalstring(context, mapping, args[1])
src = evalstring(context, mapping, args[2])
Yuya Nishihara
templater: catch regexp error at sub() function...
r26188 try:
patre = re.compile(pat)
except re.error:
# i18n: "sub" is a keyword
raise error.ParseError(_("sub got an invalid pattern: %s") % pat)
try:
yield patre.sub(rpl, src)
except re.error:
# i18n: "sub" is a keyword
raise error.ParseError(_("sub got an invalid replacement: %s") % rpl)
Alexander Plavin
templater: sort functions alphabetically, as filters are
r19390
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('startswith(pattern, text)')
Ryan McElroy
templater: introduce startswith function...
r21821 def startswith(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Returns the value from the "text" argument
Gregory Szorc
templater: add consistent docstrings to functions...
r24586 if it begins with the content from the "pattern" argument."""
Ryan McElroy
templater: introduce startswith function...
r21821 if len(args) != 2:
FUJIWARA Katsunori
templater: add i18n comments to error messages of newly added functions...
r21960 # i18n: "startswith" is a keyword
Ryan McElroy
templater: introduce startswith function...
r21821 raise error.ParseError(_("startswith expects two arguments"))
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 patn = evalstring(context, mapping, args[0])
text = evalstring(context, mapping, args[1])
Ryan McElroy
templater: introduce startswith function...
r21821 if text.startswith(patn):
return text
return ''
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 @templatefunc('word(number, text[, separator])')
Ryan McElroy
templater: introduce word function...
r21846 def word(context, mapping, args):
FUJIWARA Katsunori
templater: use templatefunc to mark a function as template function...
r28696 """Return the nth word from a string."""
Ryan McElroy
templater: introduce word function...
r21846 if not (2 <= len(args) <= 3):
FUJIWARA Katsunori
templater: add i18n comments to error messages of newly added functions...
r21960 # i18n: "word" is a keyword
Ryan McElroy
templater: introduce word function...
r21846 raise error.ParseError(_("word expects two or three arguments, got %d")
% len(args))
Yuya Nishihara
templater: factor out function that evaluates argument as integer...
r28343 num = evalinteger(context, mapping, args[0],
# i18n: "word" is a keyword
_("word expects an integer index"))
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 text = evalstring(context, mapping, args[1])
Ryan McElroy
templater: introduce word function...
r21846 if len(args) == 3:
Yuya Nishihara
templater: factor out thin helper that evaluates argument as string...
r28348 splitter = evalstring(context, mapping, args[2])
Ryan McElroy
templater: introduce word function...
r21846 else:
splitter = None
tokens = text.split(splitter)
Matt Harbison
templater: protect word() from crashing on out of range negative value...
r26502 if num >= len(tokens) or num < -len(tokens):
Ryan McElroy
templater: introduce word function...
r21846 return ''
else:
return tokens[num]
Yuya Nishihara
templater: switch methods table on compileexp() of func args and inner expr...
r25001 # methods to interpret function arguments or inner expressions (e.g. {_(x)})
exprmethods = {
Yuya Nishihara
templater: tokenize decimal integer literal (issue4638) (BC)...
r25002 "integer": lambda e, c: (runinteger, e[1]),
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 "string": lambda e, c: (runstring, e[1]),
"symbol": lambda e, c: (runsymbol, e[1]),
Yuya Nishihara
templater: take any string literals as template, but not for rawstring (BC)...
r25596 "template": buildtemplate,
Yuya Nishihara
templater: switch methods table on compileexp() of func args and inner expr...
r25001 "group": lambda e, c: compileexp(e[1], c, exprmethods),
Yuya Nishihara
templater: add dot operator to easily access a sub item...
r34536 ".": buildmember,
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 "|": buildfilter,
"%": buildmap,
"func": buildfunc,
Yuya Nishihara
templater: add parsing rule for key-value pair...
r31885 "keyvalue": buildkeyvaluepair,
Simon Farnsworth
templater: provide arithmetic operations on integers...
r30115 "+": lambda e, c: buildarithmetic(e, c, lambda a, b: a + b),
"-": lambda e, c: buildarithmetic(e, c, lambda a, b: a - b),
"negate": buildnegate,
"*": lambda e, c: buildarithmetic(e, c, lambda a, b: a * b),
"/": lambda e, c: buildarithmetic(e, c, lambda a, b: a // b),
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 }
Yuya Nishihara
templater: switch methods table on compileexp() of func args and inner expr...
r25001 # methods to interpret top-level template (e.g. {x}, {x|_}, {x % "y"})
methods = exprmethods.copy()
Yuya Nishihara
templater: tokenize decimal integer literal (issue4638) (BC)...
r25002 methods["integer"] = exprmethods["symbol"] # '{1}' as variable
Yuya Nishihara
templater: switch methods table on compileexp() of func args and inner expr...
r25001
Yuya Nishihara
templater: add parsing and expansion rules to process "templatealias" section...
r28912 class _aliasrules(parser.basealiasrules):
"""Parsing and expansion rule set of template aliases"""
_section = _('template alias')
_parse = staticmethod(_parseexpr)
@staticmethod
def _trygetfunc(tree):
"""Return (name, args) if tree is func(...) or ...|filter; otherwise
None"""
if tree[0] == 'func' and tree[1][0] == 'symbol':
return tree[1][1], getlist(tree[2])
if tree[0] == '|' and tree[2][0] == 'symbol':
return tree[2][1], [tree[1]]
def expandaliases(tree, aliases):
"""Return new tree of aliases are expanded"""
aliasmap = _aliasrules.buildmap(aliases)
return _aliasrules.expand(aliasmap, tree)
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 # template engine
Vadim Gelfer
use safer string parser for template engine.
r1901
Dirkjan Ochtman
templater: provide the standard template filters by default
r8360 stringify = templatefilters.stringify
Brendan Cully
Allow hgweb to search for templates in more than one path....
r7107
Matt Mackall
templater: use recursive flattening...
r10850 def _flatten(thing):
'''yield a single stream from a possibly nested set of iterators'''
Yuya Nishihara
templater: remove __iter__() from _hybrid, resolve it explicitly...
r31880 thing = templatekw.unwraphybrid(thing)
Pulkit Goyal
py3: replace str with bytes in isinstance()...
r32970 if isinstance(thing, bytes):
Matt Mackall
templater: use recursive flattening...
r10850 yield thing
Augie Fackler
templater: explode if we try to emit a str...
r34724 elif isinstance(thing, str):
# We can only hit this on Python 3, and it's here to guard
# against infinite recursion.
raise error.ProgrammingError('Mercurial IO including templates is done'
Augie Fackler
templater: show repr of string we're rejecting...
r36729 ' with bytes, not strings, got %r' % thing)
Yuya Nishihara
templater: make it clearer that _flatten() omits None
r29815 elif thing is None:
pass
Augie Fackler
globally: use safehasattr(x, '__iter__') instead of hasattr(x, '__iter__')
r14944 elif not util.safehasattr(thing, '__iter__'):
Pulkit Goyal
py3: use pycompat.bytestr() in place of str()
r32974 yield pycompat.bytestr(thing)
Matt Mackall
templater: raise nested functions
r10852 else:
Matt Mackall
templater: use recursive flattening...
r10850 for i in thing:
Yuya Nishihara
templater: remove __iter__() from _hybrid, resolve it explicitly...
r31880 i = templatekw.unwraphybrid(i)
Pulkit Goyal
py3: replace str with bytes in isinstance()...
r32970 if isinstance(i, bytes):
Matt Mackall
templater: use recursive flattening...
r10850 yield i
Yuya Nishihara
templater: make it clearer that _flatten() omits None
r29815 elif i is None:
pass
Augie Fackler
globally: use safehasattr(x, '__iter__') instead of hasattr(x, '__iter__')
r14944 elif not util.safehasattr(i, '__iter__'):
Pulkit Goyal
py3: use pycompat.bytestr() in place of str()
r32974 yield pycompat.bytestr(i)
Yuya Nishihara
templater: make it clearer that _flatten() omits None
r29815 else:
Matt Mackall
templater: use recursive flattening...
r10850 for j in _flatten(i):
yield j
Yuya Nishihara
templater: rename parsestring() to unquotestring() (API)...
r24988 def unquotestring(s):
Yuya Nishihara
templater: relax unquotestring() to fall back to bare string...
r28630 '''unwrap quotes if any; otherwise returns unmodified string'''
Yuya Nishihara
templater: do not strip non-quote characters from template config...
r28687 if len(s) < 2 or s[0] not in "'\"" or s[0] != s[-1]:
Yuya Nishihara
templater: relax unquotestring() to fall back to bare string...
r28630 return s
Yuya Nishihara
templater: remove workaround for escaped quoted string in quoted template...
r25696 return s[1:-1]
Vadim Gelfer
move hgweb template code out to templater
r1896
Dirkjan Ochtman
templater: separate template management and actual string processing
r8218 class engine(object):
Vadim Gelfer
add doc comments to template code.
r1909 '''template expansion engine.
template expansion works like this. a map file contains key=value
pairs. if value is quoted, it is treated as string. otherwise, it
is treated as name of template file.
templater is asked to expand a key in map. it looks up key, and
TK Soh
minor typo fix in templater's docstring
r4334 looks for strings like this: {foo}. it expands {foo} by looking up
Vadim Gelfer
add doc comments to template code.
r1909 foo in map, and substituting it. expansion is recursive: it stops
when there is no more {foo} to replace.
expansion also allows formatting and filtering.
format uses key to expand each item in list. syntax is
{key%format}.
filter uses function to transform value. syntax is
{key|filter1|filter2|...}.'''
Yuya Nishihara
templater: keep default resources per template engine (API)...
r35484 def __init__(self, loader, filters=None, defaults=None, resources=None,
aliases=()):
Matt Mackall
templater: privatize class variables
r10848 self._loader = loader
Pierre-Yves David
templater: remove a mutable default argument...
r26330 if filters is None:
filters = {}
Matt Mackall
templater: privatize class variables
r10848 self._filters = filters
Pierre-Yves David
templater: remove a mutable default argument...
r26331 if defaults is None:
defaults = {}
Yuya Nishihara
templater: keep default resources per template engine (API)...
r35484 if resources is None:
resources = {}
Matt Mackall
templater: privatize class variables
r10848 self._defaults = defaults
Yuya Nishihara
templater: keep default resources per template engine (API)...
r35484 self._resources = resources
Yuya Nishihara
templater: load and expand aliases by template engine (API) (issue4842)...
r28957 self._aliasmap = _aliasrules.buildmap(aliases)
Yuya Nishihara
templater: lift parsed and compiled templates to generic data types...
r28545 self._cache = {} # key: (func, data)
Dirkjan Ochtman
templater: separate template management and actual string processing
r8218
Yuya Nishihara
templater: look up mapping table through template engine...
r35483 def symbol(self, mapping, key):
"""Resolve symbol to value or function; None if nothing found"""
Yuya Nishihara
templater: look up symbols/resources as if they were separated (issue5699)...
r35486 v = None
if key not in self._resources:
v = mapping.get(key)
Yuya Nishihara
templater: look up mapping table through template engine...
r35483 if v is None:
v = self._defaults.get(key)
return v
def resource(self, mapping, key):
"""Return internal data (e.g. cache) used for keyword/function
evaluation"""
Yuya Nishihara
templater: look up symbols/resources as if they were separated (issue5699)...
r35486 v = None
if key in self._resources:
v = mapping.get(key)
Yuya Nishihara
templater: keep default resources per template engine (API)...
r35484 if v is None:
v = self._resources.get(key)
if v is None:
Yuya Nishihara
templater: specialize ResourceUnavailable error so that it can be caught...
r36462 raise ResourceUnavailable(_('template resource not available: %s')
% key)
Yuya Nishihara
templater: keep default resources per template engine (API)...
r35484 return v
Yuya Nishihara
templater: look up mapping table through template engine...
r35483
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 def _load(self, t):
'''load, parse, and cache a template'''
if t not in self._cache:
Yuya Nishihara
templater: abort if infinite recursion detected while compiling...
r27940 # put poison to cut recursion while compiling 't'
Yuya Nishihara
templater: lift parsed and compiled templates to generic data types...
r28545 self._cache[t] = (_runrecursivesymbol, t)
Yuya Nishihara
templater: abort if infinite recursion detected while compiling...
r27940 try:
Yuya Nishihara
templater: inline compiletemplate() function into engine...
r28956 x = parse(self._loader(t))
Yuya Nishihara
templater: load and expand aliases by template engine (API) (issue4842)...
r28957 if self._aliasmap:
x = _aliasrules.expand(self._aliasmap, x)
Yuya Nishihara
templater: inline compiletemplate() function into engine...
r28956 self._cache[t] = compileexp(x, self, methods)
Yuya Nishihara
templater: abort if infinite recursion detected while compiling...
r27940 except: # re-raises
del self._cache[t]
raise
Matt Mackall
templater: use the parser.py parser to extend the templater syntax
r13176 return self._cache[t]
Matt Mackall
templater: drop raw method
r10853 def process(self, t, mapping):
'''Perform expansion. t is name of map element to expand.
mapping contains added elements for use during expansion. Is a
generator.'''
Yuya Nishihara
templater: lift parsed and compiled templates to generic data types...
r28545 func, data = self._load(t)
return _flatten(func(self, mapping, data))
Dirkjan Ochtman
templater: separate template management and actual string processing
r8218
Dirkjan Ochtman
templater: make the templating engine pluggable to some extent
r8361 engines = {'default': engine}
Iulian Stana
templater: show the style list when I try to use a wrong one...
r19125 def stylelist():
Mads Kiilerich
templater: introduce templatepaths for getting paths searched for templates...
r22634 paths = templatepaths()
Simon Heimberg
templater: selecting a style with no templates does not crash (issue4140)...
r20312 if not paths:
return _('no templates found, try `hg debuginstall` for more info')
timeless
cleanup: remove superfluous space after space after equals (python)
r27637 dirlist = os.listdir(paths[0])
Iulian Stana
templater: show the style list when I try to use a wrong one...
r19125 stylelist = []
for file in dirlist:
split = file.split(".")
timeless
templater: ignore orig/rej files...
r28403 if split[-1] in ('orig', 'rej'):
continue
Iulian Stana
templater: show the style list when I try to use a wrong one...
r19125 if split[0] == "map-cmdline":
stylelist.append(split[1])
Augie Fackler
templater: fix output instability from gsoc patches
r19127 return ", ".join(sorted(stylelist))
Iulian Stana
templater: show the style list when I try to use a wrong one...
r19125
Yuya Nishihara
templater: extract function that loads template map file...
r28953 def _readmapfile(mapfile):
"""Load template elements from the given map file"""
if not os.path.exists(mapfile):
raise error.Abort(_("style '%s' not found") % mapfile,
hint=_("available styles: %s") % stylelist())
base = os.path.dirname(mapfile)
conf = config.config(includepaths=templatepaths())
Yuya Nishihara
templater: load template fragments from [templates] section in map file...
r34715 conf.read(mapfile, remap={'': 'templates'})
Yuya Nishihara
templater: extract function that loads template map file...
r28953
cache = {}
tmap = {}
Yuya Nishihara
templater: load aliases from [templatealias] section in map file...
r34716 aliases = []
Yuya Nishihara
templater: simplify merge of __base__ dicts by reading it first
r34713
Yuya Nishihara
templater: load template fragments from [templates] section in map file...
r34715 val = conf.get('templates', '__base__')
Yuya Nishihara
templater: simplify merge of __base__ dicts by reading it first
r34713 if val and val[0] not in "'\"":
# treat as a pointer to a base class for this style
path = util.normpath(os.path.join(base, val))
# fallback check in template paths
if not os.path.exists(path):
for p in templatepaths():
p2 = util.normpath(os.path.join(p, val))
if os.path.isfile(p2):
path = p2
break
p3 = util.normpath(os.path.join(p2, "map"))
if os.path.isfile(p3):
path = p3
break
Yuya Nishihara
templater: load aliases from [templatealias] section in map file...
r34716 cache, tmap, aliases = _readmapfile(path)
Yuya Nishihara
templater: simplify merge of __base__ dicts by reading it first
r34713
Yuya Nishihara
templater: load template fragments from [templates] section in map file...
r34715 for key, val in conf['templates'].items():
Yuya Nishihara
templater: extract function that loads template map file...
r28953 if not val:
Yuya Nishihara
templater: load template fragments from [templates] section in map file...
r34715 raise error.ParseError(_('missing value'),
conf.source('templates', key))
Yuya Nishihara
templater: extract function that loads template map file...
r28953 if val[0] in "'\"":
if val[0] != val[-1]:
raise error.ParseError(_('unmatched quotes'),
Yuya Nishihara
templater: load template fragments from [templates] section in map file...
r34715 conf.source('templates', key))
Yuya Nishihara
templater: extract function that loads template map file...
r28953 cache[key] = unquotestring(val)
Yuya Nishihara
templater: simplify merge of __base__ dicts by reading it first
r34713 elif key != '__base__':
Yuya Nishihara
templater: extract function that loads template map file...
r28953 val = 'default', val
if ':' in val[1]:
val = val[1].split(':', 1)
tmap[key] = val[0], os.path.join(base, val[1])
Yuya Nishihara
templater: load aliases from [templatealias] section in map file...
r34716 aliases.extend(conf['templatealias'].items())
return cache, tmap, aliases
Yuya Nishihara
templater: extract function that loads template map file...
r28953
Dirkjan Ochtman
templater: separate template management and actual string processing
r8218 class templater(object):
Yuya Nishihara
templater: keep default resources per template engine (API)...
r35484 def __init__(self, filters=None, defaults=None, resources=None,
cache=None, aliases=(), minchunk=1024, maxchunk=65536):
Yuya Nishihara
templater: rewrite docstring of templater.__init__()...
r35497 """Create template engine optionally with preloaded template fragments
- ``filters``: a dict of functions to transform a value into another.
- ``defaults``: a dict of symbol values/functions; may be overridden
by a ``mapping`` dict.
- ``resources``: a dict of internal data (e.g. cache), inaccessible
from user template; may be overridden by a ``mapping`` dict.
- ``cache``: a dict of preloaded template fragments.
- ``aliases``: a list of alias (name, replacement) pairs.
self.cache may be updated later to register additional template
fragments.
"""
Pierre-Yves David
templater: remove a mutable default argument...
r26332 if filters is None:
filters = {}
Pierre-Yves David
templater: remove a mutable default argument...
r26333 if defaults is None:
defaults = {}
Yuya Nishihara
templater: keep default resources per template engine (API)...
r35484 if resources is None:
resources = {}
Pierre-Yves David
templater: remove a mutable default argument...
r26334 if cache is None:
cache = {}
Shun-ichi Goto
Duplicate cache when creating templater.
r1975 self.cache = cache.copy()
Vadim Gelfer
move hgweb template code out to templater
r1896 self.map = {}
Dirkjan Ochtman
templater: provide the standard template filters by default
r8360 self.filters = templatefilters.filters.copy()
self.filters.update(filters)
Vadim Gelfer
fix template bug that made hgweb break....
r1964 self.defaults = defaults
Yuya Nishihara
templater: keep default resources per template engine (API)...
r35484 self._resources = {'templ': self}
self._resources.update(resources)
Yuya Nishihara
templater: load and expand aliases by template engine (API) (issue4842)...
r28957 self._aliases = aliases
Brendan Cully
templater: return data in increasing chunk sizes...
r7396 self.minchunk, self.maxchunk = minchunk, maxchunk
Matt Mackall
templater: clarify engine caching
r13187 self.ecache = {}
Vadim Gelfer
move hgweb template code out to templater
r1896
Yuya Nishihara
templater: separate function to create templater from map file (API)...
r28954 @classmethod
Yuya Nishihara
templater: keep default resources per template engine (API)...
r35484 def frommapfile(cls, mapfile, filters=None, defaults=None, resources=None,
cache=None, minchunk=1024, maxchunk=65536):
Yuya Nishihara
templater: separate function to create templater from map file (API)...
r28954 """Create templater from the specified map file"""
Yuya Nishihara
templater: keep default resources per template engine (API)...
r35484 t = cls(filters, defaults, resources, cache, [], minchunk, maxchunk)
Yuya Nishihara
templater: load aliases from [templatealias] section in map file...
r34716 cache, tmap, aliases = _readmapfile(mapfile)
Yuya Nishihara
templater: separate function to create templater from map file (API)...
r28954 t.cache.update(cache)
t.map = tmap
Yuya Nishihara
templater: load aliases from [templatealias] section in map file...
r34716 t._aliases = aliases
Yuya Nishihara
templater: separate function to create templater from map file (API)...
r28954 return t
Vadim Gelfer
move hgweb template code out to templater
r1896
Vadim Gelfer
many small changes to templater....
r1899 def __contains__(self, key):
Matt Mackall
templater: simplify cache and remove filter argument in __call__
r3637 return key in self.cache or key in self.map
Vadim Gelfer
many small changes to templater....
r1899
Dirkjan Ochtman
templater: separate template management and actual string processing
r8218 def load(self, t):
Dirkjan Ochtman
templater: make a template a string-only iterator
r6783 '''Get the template for the given template name. Use a local cache.'''
Brodie Rao
cleanup: "not x in y" -> "x not in y"
r16686 if t not in self.cache:
Vadim Gelfer
improve template errors when something is wrong.
r1905 try:
Dan Villiom Podlaski Christiansen
prevent transient leaks of file handle by using new helper functions...
r14168 self.cache[t] = util.readfile(self.map[t][1])
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except KeyError as inst:
Alexander Plavin
templater: support using templates with non-standard names from map file...
r19770 raise TemplateNotFound(_('"%s" not in template map') %
inst.args[0])
Gregory Szorc
global: mass rewrite to use modern exception syntax...
r25660 except IOError as inst:
Yuya Nishihara
py3: move between bytes and unicode when re-raising IOError...
r36518 reason = (_('template file %s: %s')
% (self.map[t][1], util.forcebytestr(inst.args[1])))
raise IOError(inst.args[0], encoding.strfromlocal(reason))
Dirkjan Ochtman
templater: make a template a string-only iterator
r6783 return self.cache[t]
Vadim Gelfer
move hgweb template code out to templater
r1896
Yuya Nishihara
templater: add simple interface for unnamed template (API)...
r32873 def render(self, mapping):
"""Render the default unnamed template and return result as string"""
Pulkit Goyal
py3: use pycompat.strkwargs() before passing a dict as keyword argument...
r34348 mapping = pycompat.strkwargs(mapping)
Yuya Nishihara
templater: add simple interface for unnamed template (API)...
r32873 return stringify(self('', **mapping))
Matt Mackall
templater: map -> mapping
r10847 def __call__(self, t, **mapping):
Pulkit Goyal
py3: convert kwargs keys' back to bytes using pycompat.byteskwargs()
r33016 mapping = pycompat.byteskwargs(mapping)
Dirkjan Ochtman
templater: make the templating engine pluggable to some extent
r8361 ttype = t in self.map and self.map[t][0] or 'default'
Matt Mackall
templater: clarify engine caching
r13187 if ttype not in self.ecache:
Yuya Nishihara
templater: give better error message for invalid engine type...
r28831 try:
ecls = engines[ttype]
except KeyError:
raise error.Abort(_('invalid template engine: %s') % ttype)
Yuya Nishihara
templater: load and expand aliases by template engine (API) (issue4842)...
r28957 self.ecache[ttype] = ecls(self.load, self.filters, self.defaults,
Yuya Nishihara
templater: keep default resources per template engine (API)...
r35484 self._resources, self._aliases)
Matt Mackall
templater: clarify engine caching
r13187 proc = self.ecache[ttype]
Dirkjan Ochtman
templater: make the templating engine pluggable to some extent
r8361
Matt Mackall
templater: map -> mapping
r10847 stream = proc.process(t, mapping)
Brendan Cully
templater: return data in increasing chunk sizes...
r7396 if self.minchunk:
stream = util.increasingchunks(stream, min=self.minchunk,
max=self.maxchunk)
return stream
Dirkjan Ochtman
kill some trailing spaces
r7434
Mads Kiilerich
templater: introduce templatepaths for getting paths searched for templates...
r22634 def templatepaths():
'''return locations used for template files.'''
Mads Kiilerich
templater: don't search randomly for templates - trust util.datapath...
r22636 pathsrel = ['templates']
Mads Kiilerich
templater: inline global 'path' list in templatepaths
r22635 paths = [os.path.normpath(os.path.join(util.datapath, f))
for f in pathsrel]
return [p for p in paths if os.path.isdir(p)]
Vadim Gelfer
move changeset_templater into templater module.
r2189
Mads Kiilerich
templater: introduce templatepaths for getting paths searched for templates...
r22634 def templatepath(name):
'''return location of template file. returns None if not found.'''
for p in templatepaths():
f = os.path.join(p, name)
if os.path.exists(f):
return f
return None
Vadim Gelfer
move changeset_templater into templater module.
r2189
Dirkjan Ochtman
hgweb: don't choke when an inexistent style is requested (issue1901)
r9842 def stylemap(styles, paths=None):
Dirkjan Ochtman
templater: move stylemap function from hgweb to templater
r7966 """Return path to mapfile for a given style.
Searches mapfile in the following locations:
1. templatepath/style/map
2. templatepath/map-style
3. templatepath/map
"""
if paths is None:
Mads Kiilerich
templater: introduce templatepaths for getting paths searched for templates...
r22634 paths = templatepaths()
Yuya Nishihara
py3: replace type 'str' by 'bytes' in templater.py
r36568 elif isinstance(paths, bytes):
Dirkjan Ochtman
templater: fix little problem from stylemap() changes
r8223 paths = [paths]
Dirkjan Ochtman
templater: move stylemap function from hgweb to templater
r7966
Yuya Nishihara
py3: replace type 'str' by 'bytes' in templater.py
r36568 if isinstance(styles, bytes):
Dirkjan Ochtman
hgweb: don't choke when an inexistent style is requested (issue1901)
r9842 styles = [styles]
for style in styles:
Yuya Nishihara
hgweb: prevent loading style map from directories other than specified paths...
r24296 # only plain name is allowed to honor template paths
if (not style
Yuya Nishihara
py3: make os.curdir a bytes
r36666 or style in (pycompat.oscurdir, pycompat.ospardir)
Pulkit Goyal
py3: replace os.sep with pycompat.ossep (part 3 of 4)
r30615 or pycompat.ossep in style
Pulkit Goyal
py3: replace os.altsep with pycompat.altsep...
r30625 or pycompat.osaltsep and pycompat.osaltsep in style):
Dirkjan Ochtman
hgweb: don't choke when an inexistent style is requested (issue1901)
r9842 continue
locations = [os.path.join(style, 'map'), 'map-' + style]
locations.append('map')
for path in paths:
for location in locations:
mapfile = os.path.join(path, location)
if os.path.isfile(mapfile):
return style, mapfile
Dirkjan Ochtman
templater: move stylemap function from hgweb to templater
r7966
raise RuntimeError("No hgweb templates found in %r" % paths)
Yuya Nishihara
templater: tell hggettext to collect help of template functions
r24601
FUJIWARA Katsunori
registrar: add templatefunc to mark a function as template function (API)...
r28695 def loadfunction(ui, extname, registrarobj):
"""Load template function from specified registrarobj
"""
for name, func in registrarobj._table.iteritems():
funcs[name] = func
Yuya Nishihara
templater: tell hggettext to collect help of template functions
r24601 # tell hggettext to extract docstrings from these functions:
i18nfunctions = funcs.values()