templateutil.py
503 lines
| 16.7 KiB
| text/x-python
|
PythonLexer
/ mercurial / templateutil.py
Yuya Nishihara
|
r36931 | # templateutil.py - utility for template evaluation | ||
# | ||||
# Copyright 2005, 2006 Matt Mackall <mpm@selenic.com> | ||||
# | ||||
# This software may be used and distributed according to the terms of the | ||||
# GNU General Public License version 2 or any later version. | ||||
from __future__ import absolute_import | ||||
import types | ||||
from .i18n import _ | ||||
from . import ( | ||||
error, | ||||
pycompat, | ||||
util, | ||||
) | ||||
Yuya Nishihara
|
r37102 | from .utils import ( | ||
Yuya Nishihara
|
r37241 | dateutil, | ||
Yuya Nishihara
|
r37102 | stringutil, | ||
) | ||||
Yuya Nishihara
|
r36931 | |||
class ResourceUnavailable(error.Abort): | ||||
pass | ||||
class TemplateNotFound(error.Abort): | ||||
pass | ||||
Yuya Nishihara
|
r37244 | # stub for representing a date type; may be a real date type that can | ||
# provide a readable string value | ||||
class date(object): | ||||
pass | ||||
Yuya Nishihara
|
r36939 | class hybrid(object): | ||
"""Wrapper for list or dict to support legacy template | ||||
This class allows us to handle both: | ||||
- "{files}" (legacy command-line-specific list hack) and | ||||
- "{files % '{file}\n'}" (hgweb-style with inlining and function support) | ||||
and to access raw values: | ||||
- "{ifcontains(file, files, ...)}", "{ifcontains(key, extras, ...)}" | ||||
- "{get(extras, key)}" | ||||
- "{files|json}" | ||||
""" | ||||
def __init__(self, gen, values, makemap, joinfmt, keytype=None): | ||||
if gen is not None: | ||||
self.gen = gen # generator or function returning generator | ||||
self._values = values | ||||
self._makemap = makemap | ||||
self.joinfmt = joinfmt | ||||
self.keytype = keytype # hint for 'x in y' where type(x) is unresolved | ||||
def gen(self): | ||||
"""Default generator to stringify this as {join(self, ' ')}""" | ||||
for i, x in enumerate(self._values): | ||||
if i > 0: | ||||
yield ' ' | ||||
yield self.joinfmt(x) | ||||
def itermaps(self): | ||||
makemap = self._makemap | ||||
for x in self._values: | ||||
yield makemap(x) | ||||
def __contains__(self, x): | ||||
return x in self._values | ||||
def __getitem__(self, key): | ||||
return self._values[key] | ||||
def __len__(self): | ||||
return len(self._values) | ||||
def __iter__(self): | ||||
return iter(self._values) | ||||
def __getattr__(self, name): | ||||
if name not in (r'get', r'items', r'iteritems', r'iterkeys', | ||||
r'itervalues', r'keys', r'values'): | ||||
raise AttributeError(name) | ||||
return getattr(self._values, name) | ||||
class mappable(object): | ||||
"""Wrapper for non-list/dict object to support map operation | ||||
This class allows us to handle both: | ||||
- "{manifest}" | ||||
- "{manifest % '{rev}:{node}'}" | ||||
- "{manifest.rev}" | ||||
Unlike a hybrid, this does not simulate the behavior of the underling | ||||
Yuya Nishihara
|
r37180 | value. Use unwrapvalue(), unwrapastype(), or unwraphybrid() to obtain | ||
the inner object. | ||||
Yuya Nishihara
|
r36939 | """ | ||
def __init__(self, gen, key, value, makemap): | ||||
if gen is not None: | ||||
self.gen = gen # generator or function returning generator | ||||
self._key = key | ||||
self._value = value # may be generator of strings | ||||
self._makemap = makemap | ||||
def gen(self): | ||||
yield pycompat.bytestr(self._value) | ||||
def tomap(self): | ||||
return self._makemap(self._key) | ||||
def itermaps(self): | ||||
yield self.tomap() | ||||
def hybriddict(data, key='key', value='value', fmt=None, gen=None): | ||||
"""Wrap data to support both dict-like and string-like operations""" | ||||
prefmt = pycompat.identity | ||||
if fmt is None: | ||||
fmt = '%s=%s' | ||||
prefmt = pycompat.bytestr | ||||
return hybrid(gen, data, lambda k: {key: k, value: data[k]}, | ||||
lambda k: fmt % (prefmt(k), prefmt(data[k]))) | ||||
def hybridlist(data, name, fmt=None, gen=None): | ||||
"""Wrap data to support both list-like and string-like operations""" | ||||
prefmt = pycompat.identity | ||||
if fmt is None: | ||||
fmt = '%s' | ||||
prefmt = pycompat.bytestr | ||||
return hybrid(gen, data, lambda x: {name: x}, lambda x: fmt % prefmt(x)) | ||||
def unwraphybrid(thing): | ||||
"""Return an object which can be stringified possibly by using a legacy | ||||
template""" | ||||
gen = getattr(thing, 'gen', None) | ||||
if gen is None: | ||||
return thing | ||||
if callable(gen): | ||||
return gen() | ||||
return gen | ||||
def unwrapvalue(thing): | ||||
"""Move the inner value object out of the wrapper""" | ||||
if not util.safehasattr(thing, '_value'): | ||||
return thing | ||||
return thing._value | ||||
def wraphybridvalue(container, key, value): | ||||
"""Wrap an element of hybrid container to be mappable | ||||
The key is passed to the makemap function of the given container, which | ||||
should be an item generated by iter(container). | ||||
""" | ||||
makemap = getattr(container, '_makemap', None) | ||||
if makemap is None: | ||||
return value | ||||
if util.safehasattr(value, '_makemap'): | ||||
# a nested hybrid list/dict, which has its own way of map operation | ||||
return value | ||||
return mappable(None, key, value, makemap) | ||||
def compatdict(context, mapping, name, data, key='key', value='value', | ||||
fmt=None, plural=None, separator=' '): | ||||
"""Wrap data like hybriddict(), but also supports old-style list template | ||||
This exists for backward compatibility with the old-style template. Use | ||||
hybriddict() for new template keywords. | ||||
""" | ||||
c = [{key: k, value: v} for k, v in data.iteritems()] | ||||
Yuya Nishihara
|
r37086 | f = _showcompatlist(context, mapping, name, c, plural, separator) | ||
Yuya Nishihara
|
r36939 | return hybriddict(data, key=key, value=value, fmt=fmt, gen=f) | ||
def compatlist(context, mapping, name, data, element=None, fmt=None, | ||||
plural=None, separator=' '): | ||||
"""Wrap data like hybridlist(), but also supports old-style list template | ||||
This exists for backward compatibility with the old-style template. Use | ||||
hybridlist() for new template keywords. | ||||
""" | ||||
Yuya Nishihara
|
r37086 | f = _showcompatlist(context, mapping, name, data, plural, separator) | ||
Yuya Nishihara
|
r36939 | return hybridlist(data, name=element or name, fmt=fmt, gen=f) | ||
Yuya Nishihara
|
r37086 | def _showcompatlist(context, mapping, name, values, plural=None, separator=' '): | ||
"""Return a generator that renders old-style list template | ||||
Yuya Nishihara
|
r36939 | name is name of key in template map. | ||
values is list of strings or dicts. | ||||
plural is plural of name, if not simply name + 's'. | ||||
separator is used to join values as a string | ||||
expansion works like this, given name 'foo'. | ||||
if values is empty, expand 'no_foos'. | ||||
if 'foo' not in template map, return values as a string, | ||||
joined by 'separator'. | ||||
expand 'start_foos'. | ||||
for each value, expand 'foo'. if 'last_foo' in template | ||||
map, expand it instead of 'foo' for last key. | ||||
expand 'end_foos'. | ||||
Yuya Nishihara
|
r37086 | """ | ||
Yuya Nishihara
|
r36939 | if not plural: | ||
plural = name + 's' | ||||
if not values: | ||||
noname = 'no_' + plural | ||||
Yuya Nishihara
|
r37086 | if context.preload(noname): | ||
yield context.process(noname, mapping) | ||||
Yuya Nishihara
|
r36939 | return | ||
Yuya Nishihara
|
r37086 | if not context.preload(name): | ||
Yuya Nishihara
|
r36939 | if isinstance(values[0], bytes): | ||
yield separator.join(values) | ||||
else: | ||||
for v in values: | ||||
r = dict(v) | ||||
r.update(mapping) | ||||
yield r | ||||
return | ||||
startname = 'start_' + plural | ||||
Yuya Nishihara
|
r37086 | if context.preload(startname): | ||
yield context.process(startname, mapping) | ||||
Yuya Nishihara
|
r36939 | def one(v, tag=name): | ||
Yuya Nishihara
|
r37092 | vmapping = {} | ||
Yuya Nishihara
|
r36939 | try: | ||
vmapping.update(v) | ||||
# Python 2 raises ValueError if the type of v is wrong. Python | ||||
# 3 raises TypeError. | ||||
except (AttributeError, TypeError, ValueError): | ||||
try: | ||||
# Python 2 raises ValueError trying to destructure an e.g. | ||||
# bytes. Python 3 raises TypeError. | ||||
for a, b in v: | ||||
vmapping[a] = b | ||||
except (TypeError, ValueError): | ||||
vmapping[name] = v | ||||
Yuya Nishihara
|
r37092 | vmapping = context.overlaymap(mapping, vmapping) | ||
Yuya Nishihara
|
r37086 | return context.process(tag, vmapping) | ||
Yuya Nishihara
|
r36939 | lastname = 'last_' + name | ||
Yuya Nishihara
|
r37086 | if context.preload(lastname): | ||
Yuya Nishihara
|
r36939 | last = values.pop() | ||
else: | ||||
last = None | ||||
for v in values: | ||||
yield one(v) | ||||
if last is not None: | ||||
yield one(last, tag=lastname) | ||||
endname = 'end_' + plural | ||||
Yuya Nishihara
|
r37086 | if context.preload(endname): | ||
yield context.process(endname, mapping) | ||||
Yuya Nishihara
|
r36939 | |||
Yuya Nishihara
|
r37174 | def flatten(thing): | ||
"""Yield a single stream from a possibly nested set of iterators""" | ||||
thing = unwraphybrid(thing) | ||||
if isinstance(thing, bytes): | ||||
yield thing | ||||
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' | ||||
' with bytes, not strings, got %r' % thing) | ||||
elif thing is None: | ||||
pass | ||||
elif not util.safehasattr(thing, '__iter__'): | ||||
yield pycompat.bytestr(thing) | ||||
else: | ||||
for i in thing: | ||||
i = unwraphybrid(i) | ||||
if isinstance(i, bytes): | ||||
yield i | ||||
elif i is None: | ||||
pass | ||||
elif not util.safehasattr(i, '__iter__'): | ||||
yield pycompat.bytestr(i) | ||||
else: | ||||
for j in flatten(i): | ||||
yield j | ||||
Yuya Nishihara
|
r36938 | def stringify(thing): | ||
"""Turn values into bytes by converting into text and concatenating them""" | ||||
Yuya Nishihara
|
r37175 | if isinstance(thing, bytes): | ||
return thing # retain localstr to be round-tripped | ||||
return b''.join(flatten(thing)) | ||||
Yuya Nishihara
|
r36938 | |||
Yuya Nishihara
|
r36931 | 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 | ||||
def evalrawexp(context, mapping, arg): | ||||
"""Evaluate given argument as a bare template object which may require | ||||
further processing (such as folding generator of strings)""" | ||||
func, data = arg | ||||
return func(context, mapping, data) | ||||
def evalfuncarg(context, mapping, arg): | ||||
"""Evaluate given argument as value type""" | ||||
Yuya Nishihara
|
r37178 | return _unwrapvalue(evalrawexp(context, mapping, arg)) | ||
# TODO: unify this with unwrapvalue() once the bug of templatefunc.join() | ||||
# is fixed. we can't do that right now because join() has to take a generator | ||||
# of byte strings as it is, not a lazy byte string. | ||||
def _unwrapvalue(thing): | ||||
Yuya Nishihara
|
r36939 | thing = unwrapvalue(thing) | ||
Yuya Nishihara
|
r36931 | # evalrawexp() may return string, generator of strings or arbitrary object | ||
# such as date tuple, but filter does not want generator. | ||||
if isinstance(thing, types.GeneratorType): | ||||
thing = stringify(thing) | ||||
return thing | ||||
def evalboolean(context, mapping, arg): | ||||
"""Evaluate given argument as boolean, but also takes boolean literals""" | ||||
func, data = arg | ||||
if func is runsymbol: | ||||
thing = func(context, mapping, data, default=None) | ||||
if thing is None: | ||||
# not a template keyword, takes as a boolean literal | ||||
Yuya Nishihara
|
r37102 | thing = stringutil.parsebool(data) | ||
Yuya Nishihara
|
r36931 | else: | ||
thing = func(context, mapping, data) | ||||
Yuya Nishihara
|
r36939 | thing = unwrapvalue(thing) | ||
Yuya Nishihara
|
r36931 | 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
|
r37241 | def evaldate(context, mapping, arg, err=None): | ||
"""Evaluate given argument as a date tuple or a date string; returns | ||||
a (unixtime, offset) tuple""" | ||||
return unwrapdate(evalrawexp(context, mapping, arg), err) | ||||
def unwrapdate(thing, err=None): | ||||
thing = _unwrapvalue(thing) | ||||
try: | ||||
return dateutil.parsedate(thing) | ||||
except AttributeError: | ||||
raise error.ParseError(err or _('not a date tuple nor a string')) | ||||
Yuya Nishihara
|
r37242 | except error.ParseError: | ||
if not err: | ||||
raise | ||||
raise error.ParseError(err) | ||||
Yuya Nishihara
|
r37241 | |||
Yuya Nishihara
|
r36931 | def evalinteger(context, mapping, arg, err=None): | ||
Yuya Nishihara
|
r37179 | return unwrapinteger(evalrawexp(context, mapping, arg), err) | ||
def unwrapinteger(thing, err=None): | ||||
thing = _unwrapvalue(thing) | ||||
Yuya Nishihara
|
r36931 | try: | ||
Yuya Nishihara
|
r37179 | return int(thing) | ||
Yuya Nishihara
|
r36931 | except (TypeError, ValueError): | ||
raise error.ParseError(err or _('not an integer')) | ||||
def evalstring(context, mapping, arg): | ||||
return stringify(evalrawexp(context, mapping, arg)) | ||||
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
|
r37180 | _unwrapfuncbytype = { | ||
Yuya Nishihara
|
r37239 | None: _unwrapvalue, | ||
Yuya Nishihara
|
r37180 | bytes: stringify, | ||
Yuya Nishihara
|
r37244 | date: unwrapdate, | ||
Yuya Nishihara
|
r37180 | int: unwrapinteger, | ||
Yuya Nishihara
|
r36931 | } | ||
Yuya Nishihara
|
r37180 | def unwrapastype(thing, typ): | ||
"""Move the inner value object out of the wrapper and coerce its type""" | ||||
Yuya Nishihara
|
r36931 | try: | ||
Yuya Nishihara
|
r37180 | f = _unwrapfuncbytype[typ] | ||
Yuya Nishihara
|
r36931 | except KeyError: | ||
raise error.ProgrammingError('invalid type specified: %r' % typ) | ||||
Yuya Nishihara
|
r37180 | return f(thing) | ||
Yuya Nishihara
|
r36931 | |||
def runinteger(context, mapping, data): | ||||
return int(data) | ||||
def runstring(context, mapping, data): | ||||
return data | ||||
def _recursivesymbolblocker(key): | ||||
def showrecursion(**args): | ||||
raise error.Abort(_("recursive reference '%s' in template") % key) | ||||
return showrecursion | ||||
def runsymbol(context, mapping, key, default=''): | ||||
v = context.symbol(mapping, key) | ||||
if v is None: | ||||
# 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) | ||||
try: | ||||
v = context.process(key, safemapping) | ||||
except TemplateNotFound: | ||||
v = default | ||||
if callable(v) and getattr(v, '_requires', None) is None: | ||||
# old templatekw: expand all keywords and resources | ||||
Yuya Nishihara
|
r37088 | # (TODO: deprecate this after porting web template keywords to new API) | ||
Yuya Nishihara
|
r37091 | props = {k: context._resources.lookup(context, mapping, k) | ||
for k in context._resources.knownkeys()} | ||||
Yuya Nishihara
|
r37088 | # pass context to _showcompatlist() through templatekw._showlist() | ||
props['templ'] = context | ||||
Yuya Nishihara
|
r36931 | props.update(mapping) | ||
return v(**pycompat.strkwargs(props)) | ||||
if callable(v): | ||||
# new templatekw | ||||
try: | ||||
return v(context, mapping) | ||||
except ResourceUnavailable: | ||||
# unsupported keyword is mapped to empty just like unknown keyword | ||||
return None | ||||
return v | ||||
def runtemplate(context, mapping, template): | ||||
for arg in template: | ||||
yield evalrawexp(context, mapping, arg) | ||||
def runfilter(context, mapping, data): | ||||
arg, filt = data | ||||
Yuya Nishihara
|
r37239 | thing = evalrawexp(context, mapping, arg) | ||
Yuya Nishihara
|
r36931 | try: | ||
Yuya Nishihara
|
r37239 | thing = unwrapastype(thing, getattr(filt, '_intype', None)) | ||
Yuya Nishihara
|
r36931 | return filt(thing) | ||
Yuya Nishihara
|
r37243 | except error.ParseError as e: | ||
raise error.ParseError(bytes(e), hint=_formatfiltererror(arg, filt)) | ||||
def _formatfiltererror(arg, filt): | ||||
fn = pycompat.sysbytes(filt.__name__) | ||||
sym = findsymbolicname(arg) | ||||
if not sym: | ||||
return _("incompatible use of template filter '%s'") % fn | ||||
return (_("template filter '%s' is not compatible with keyword '%s'") | ||||
% (fn, sym)) | ||||
Yuya Nishihara
|
r36931 | |||
def runmap(context, mapping, data): | ||||
darg, targ = data | ||||
d = evalrawexp(context, mapping, darg) | ||||
if util.safehasattr(d, 'itermaps'): | ||||
diter = d.itermaps() | ||||
else: | ||||
try: | ||||
diter = iter(d) | ||||
except TypeError: | ||||
sym = findsymbolicname(darg) | ||||
if sym: | ||||
raise error.ParseError(_("keyword '%s' is not iterable") % sym) | ||||
else: | ||||
raise error.ParseError(_("%r is not iterable") % d) | ||||
for i, v in enumerate(diter): | ||||
if isinstance(v, dict): | ||||
Yuya Nishihara
|
r37092 | lm = context.overlaymap(mapping, v) | ||
lm['index'] = i | ||||
Yuya Nishihara
|
r36931 | yield evalrawexp(context, lm, targ) | ||
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. | ||||
yield v | ||||
def runmember(context, mapping, data): | ||||
darg, memb = data | ||||
d = evalrawexp(context, mapping, darg) | ||||
if util.safehasattr(d, 'tomap'): | ||||
Yuya Nishihara
|
r37092 | lm = context.overlaymap(mapping, d.tomap()) | ||
Yuya Nishihara
|
r36931 | return runsymbol(context, lm, memb) | ||
if util.safehasattr(d, 'get'): | ||||
return getdictitem(d, memb) | ||||
sym = findsymbolicname(darg) | ||||
if sym: | ||||
raise error.ParseError(_("keyword '%s' has no member") % sym) | ||||
else: | ||||
raise error.ParseError(_("%r has no member") % pycompat.bytestr(d)) | ||||
def runnegate(context, mapping, data): | ||||
data = evalinteger(context, mapping, data, | ||||
_('negation needs an integer argument')) | ||||
return -data | ||||
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')) | ||||
try: | ||||
return func(left, right) | ||||
except ZeroDivisionError: | ||||
raise error.Abort(_('division by zero is not defined')) | ||||
def getdictitem(dictarg, key): | ||||
val = dictarg.get(key) | ||||
if val is None: | ||||
return | ||||
Yuya Nishihara
|
r36939 | return wraphybridvalue(dictarg, key, val) | ||