##// END OF EJS Templates
merge: minor simplification
merge: minor simplification

File last commit:

r2649:3b53fa20 default
r2842:06c05c67 default
Show More
templater.py
532 lines | 17.1 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>
#
# This software may be used and distributed according to the terms
# of the GNU General Public License, incorporated herein by reference.
Vadim Gelfer
move hgweb template code out to templater
r1896 from demandload import demandload
Vadim Gelfer
use safer string parser for template engine.
r1901 from i18n import gettext as _
Vadim Gelfer
move changeset_templater into templater module.
r2189 from node import *
Vadim Gelfer
add changelog style to command line template....
r1987 demandload(globals(), "cStringIO cgi re sys os time urllib util textwrap")
Vadim Gelfer
use safer string parser for template engine.
r1901
esctable = {
'\\': '\\',
'r': '\r',
't': '\t',
'n': '\n',
'v': '\v',
}
Vadim Gelfer
make parsestring work with strings that do not have quotes.
r1902 def parsestring(s, quoted=True):
Vadim Gelfer
add doc comments to template code.
r1909 '''parse a string using simple c-like syntax.
string must be in quotes if quoted is True.'''
Vadim Gelfer
use safer string parser for template engine.
r1901 fp = cStringIO.StringIO()
Vadim Gelfer
make parsestring work with strings that do not have quotes.
r1902 if quoted:
first = s[0]
if len(s) < 2: raise SyntaxError(_('string too short'))
if first not in "'\"": raise SyntaxError(_('invalid quote'))
if s[-1] != first: raise SyntaxError(_('unmatched quotes'))
s = s[1:-1]
Vadim Gelfer
use safer string parser for template engine.
r1901 escape = False
Vadim Gelfer
make parsestring work with strings that do not have quotes.
r1902 for c in s:
Vadim Gelfer
use safer string parser for template engine.
r1901 if escape:
fp.write(esctable.get(c, c))
escape = False
elif c == '\\': escape = True
Vadim Gelfer
make parsestring work with strings that do not have quotes.
r1902 elif quoted and c == first: raise SyntaxError(_('string ends early'))
Vadim Gelfer
use safer string parser for template engine.
r1901 else: fp.write(c)
if escape: raise SyntaxError(_('unterminated escape'))
return fp.getvalue()
Vadim Gelfer
move hgweb template code out to templater
r1896
class templater(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
looks for atrings like this: {foo}. it expands {foo} by looking up
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|...}.'''
Vadim Gelfer
fix template bug that made hgweb break....
r1964 def __init__(self, mapfile, filters={}, defaults={}, cache={}):
Vadim Gelfer
add doc comments to template code.
r1909 '''set up template engine.
mapfile is name of file to read map definitions from.
filters is dict of functions. each transforms a value into another.
defaults is dict of default map definitions.'''
Vadim Gelfer
improve template errors when something is wrong.
r1905 self.mapfile = mapfile or 'template'
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 = {}
Vadim Gelfer
improve template errors when something is wrong.
r1905 self.base = (mapfile and os.path.dirname(mapfile)) or ''
Vadim Gelfer
move hgweb template code out to templater
r1896 self.filters = filters
Vadim Gelfer
fix template bug that made hgweb break....
r1964 self.defaults = defaults
Vadim Gelfer
move hgweb template code out to templater
r1896
Vadim Gelfer
improve template errors when something is wrong.
r1905 if not mapfile:
return
Vadim Gelfer
use safer string parser for template engine.
r1901 i = 0
Vadim Gelfer
move hgweb template code out to templater
r1896 for l in file(mapfile):
Vadim Gelfer
make --style=compact look for map-cmdline.compact....
r1914 l = l.strip()
Vadim Gelfer
use safer string parser for template engine.
r1901 i += 1
Vadim Gelfer
make --style=compact look for map-cmdline.compact....
r1914 if not l or l[0] in '#;': continue
Vadim Gelfer
improve template errors when something is wrong.
r1905 m = re.match(r'([a-zA-Z_][a-zA-Z0-9_]*)\s*=\s*(.+)$', l)
Vadim Gelfer
move hgweb template code out to templater
r1896 if m:
Vadim Gelfer
improve template errors when something is wrong.
r1905 key, val = m.groups()
if val[0] in "'\"":
try:
self.cache[key] = parsestring(val)
except SyntaxError, inst:
raise SyntaxError('%s:%s: %s' %
(mapfile, i, inst.args[0]))
else:
self.map[key] = os.path.join(self.base, val)
Vadim Gelfer
move hgweb template code out to templater
r1896 else:
Vadim Gelfer
improve template errors when something is wrong.
r1905 raise SyntaxError(_("%s:%s: parse error") % (mapfile, i))
Vadim Gelfer
move hgweb template code out to templater
r1896
Vadim Gelfer
many small changes to templater....
r1899 def __contains__(self, key):
return key in self.cache
Vadim Gelfer
move hgweb template code out to templater
r1896 def __call__(self, t, **map):
Vadim Gelfer
add doc comments to template code.
r1909 '''perform expansion.
t is name of map element to expand.
map is added elements to use during expansion.'''
Vadim Gelfer
move hgweb template code out to templater
r1896 m = self.defaults.copy()
m.update(map)
try:
tmpl = self.cache[t]
except KeyError:
Vadim Gelfer
improve template errors when something is wrong.
r1905 try:
tmpl = self.cache[t] = file(self.map[t]).read()
except IOError, inst:
raise IOError(inst.args[0], _('template file %s: %s') %
(self.map[t], inst.args[1]))
Vadim Gelfer
move hgweb template code out to templater
r1896 return self.template(tmpl, self.filters, **m)
Vadim Gelfer
improve templating....
r1904 template_re = re.compile(r"[#{]([a-zA-Z_][a-zA-Z0-9_]*)"
Vadim Gelfer
move repeated work out of inner loops.
r1900 r"((%[a-zA-Z_][a-zA-Z0-9_]*)*)"
Vadim Gelfer
improve templating....
r1904 r"((\|[a-zA-Z_][a-zA-Z0-9_]*)*)[#}]")
Vadim Gelfer
move repeated work out of inner loops.
r1900
Vadim Gelfer
move hgweb template code out to templater
r1896 def template(self, tmpl, filters={}, **map):
Vadim Gelfer
move repeated work out of inner loops.
r1900 lm = map.copy()
Vadim Gelfer
move hgweb template code out to templater
r1896 while tmpl:
Vadim Gelfer
move repeated work out of inner loops.
r1900 m = self.template_re.search(tmpl)
Vadim Gelfer
move hgweb template code out to templater
r1896 if m:
Vadim Gelfer
improve template errors when something is wrong.
r1905 start, end = m.span(0)
s, e = tmpl[start], tmpl[end - 1]
key = m.group(1)
if ((s == '#' and e != '#') or (s == '{' and e != '}')):
raise SyntaxError(_("'%s'/'%s' mismatch expanding '%s'") %
(s, e, key))
Vadim Gelfer
move repeated work out of inner loops.
r1900 if start:
yield tmpl[:start]
Vadim Gelfer
improve template errors when something is wrong.
r1905 v = map.get(key, "")
Vadim Gelfer
move hgweb template code out to templater
r1896 v = callable(v) and v(**map) or v
format = m.group(2)
fl = m.group(4)
if format:
q = v.__iter__
for i in q():
lm.update(i)
yield self(format[1:], **lm)
v = ""
elif fl:
for f in fl.split("|")[1:]:
v = filters[f](v)
yield v
Vadim Gelfer
improve template errors when something is wrong.
r1905 tmpl = tmpl[end:]
Vadim Gelfer
move hgweb template code out to templater
r1896 else:
yield tmpl
Vadim Gelfer
move repeated work out of inner loops.
r1900 break
agescales = [("second", 1),
("minute", 60),
("hour", 3600),
("day", 3600 * 24),
("week", 3600 * 24 * 7),
("month", 3600 * 24 * 30),
("year", 3600 * 24 * 365)]
agescales.reverse()
Vadim Gelfer
move hgweb template code out to templater
r1896
Vadim Gelfer
add doc comments to template code.
r1909 def age(date):
'''turn a (timestamp, tzoff) tuple into an age string.'''
Vadim Gelfer
move hgweb template code out to templater
r1896 def plural(t, c):
if c == 1:
return t
return t + "s"
def fmt(t, c):
return "%d %s" % (c, plural(t, c))
now = time.time()
Vadim Gelfer
add doc comments to template code.
r1909 then = date[0]
Vadim Gelfer
move hgweb template code out to templater
r1896 delta = max(1, int(now - then))
Vadim Gelfer
move repeated work out of inner loops.
r1900 for t, s in agescales:
Vadim Gelfer
move hgweb template code out to templater
r1896 n = delta / s
if n >= 2 or s == 1:
return fmt(t, n)
Vadim Gelfer
add changelog style to command line template....
r1987 def stringify(thing):
'''turn nested template iterator into string.'''
cs = cStringIO.StringIO()
def walk(things):
for t in things:
if hasattr(t, '__iter__'):
walk(t)
else:
cs.write(t)
walk(thing)
return cs.getvalue()
Vadim Gelfer
use demandload more.
r2470 para_re = None
space_re = None
Vadim Gelfer
add changelog style to command line template....
r1987
def fill(text, width):
'''fill many paragraphs.'''
Vadim Gelfer
use demandload more.
r2470 global para_re, space_re
if para_re is None:
para_re = re.compile('(\n\n|\n\\s*[-*]\\s*)', re.M)
space_re = re.compile(r' +')
Eric Hopper
Arrange for old copies of CGI scripts to still work.
r2535
Vadim Gelfer
add changelog style to command line template....
r1987 def findparas():
start = 0
while True:
m = para_re.search(text, start)
if not m:
w = len(text)
while w > start and text[w-1].isspace(): w -= 1
yield text[start:w], text[w:]
break
yield text[start:m.start(0)], m.group(1)
start = m.end(1)
Vadim Gelfer
move changeset_templater into templater module.
r2189
Vadim Gelfer
add changelog style to command line template....
r1987 fp = cStringIO.StringIO()
for para, rest in findparas():
fp.write(space_re.sub(' ', textwrap.fill(para, width)))
fp.write(rest)
return fp.getvalue()
Brendan Cully
templater.firstline should handle empty strings
r2559 def firstline(text):
'''return the first line of text'''
try:
return text.splitlines(1)[0].rstrip('\r\n')
except IndexError:
return ''
Vadim Gelfer
add iso date template filter.
r1906 def isodate(date):
Vadim Gelfer
add changelog style to command line template....
r1987 '''turn a (timestamp, tzoff) tuple into an iso 8631 date and time.'''
Vadim Gelfer
add iso date template filter.
r1906 return util.datestr(date, format='%Y-%m-%d %H:%M')
Benoit Boissinot
add a new template function 'hgdate'...
r2519 def hgdate(date):
'''turn a (timestamp, tzoff) tuple into an hg cset timestamp.'''
return "%d %d" % date
Vadim Gelfer
move hgweb template code out to templater
r1896 def nl2br(text):
Vadim Gelfer
add doc comments to template code.
r1909 '''replace raw newlines with xhtml line breaks.'''
Vadim Gelfer
move hgweb template code out to templater
r1896 return text.replace('\n', '<br/>\n')
def obfuscate(text):
Brendan Cully
Preserve multibyte UTF-8 characters when obfuscating....
r2649 text = unicode(text, 'utf-8', 'replace')
Vadim Gelfer
move hgweb template code out to templater
r1896 return ''.join(['&#%d;' % ord(c) for c in text])
Vadim Gelfer
improve templating....
r1904 def domain(author):
Vadim Gelfer
add doc comments to template code.
r1909 '''get domain of author, or empty string if none.'''
Vadim Gelfer
improve templating....
r1904 f = author.find('@')
if f == -1: return ''
author = author[f+1:]
f = author.find('>')
if f >= 0: author = author[:f]
return author
Vadim Gelfer
add changelog style to command line template....
r1987 def email(author):
'''get email of author.'''
r = author.find('>')
if r == -1: r = None
return author[author.find('<')+1:r]
Vadim Gelfer
move changeset_templater into templater module.
r2189
Vadim Gelfer
improve templating....
r1904 def person(author):
Vadim Gelfer
add doc comments to template code.
r1909 '''get name of author, or else username.'''
Vadim Gelfer
improve templating....
r1904 f = author.find('<')
if f == -1: return util.shortuser(author)
return author[:f].rstrip()
Vadim Gelfer
add changelog style to command line template....
r1987 def shortdate(date):
'''turn (timestamp, tzoff) tuple into iso 8631 date.'''
return util.datestr(date, format='%Y-%m-%d', timezone=False)
def indent(text, prefix):
'''indent each non-empty line of text after first with prefix.'''
fp = cStringIO.StringIO()
lines = text.splitlines()
num_lines = len(lines)
for i in xrange(num_lines):
l = lines[i]
if i and l.strip(): fp.write(prefix)
fp.write(l)
if i < num_lines - 1 or text.endswith('\n'):
fp.write('\n')
return fp.getvalue()
Vadim Gelfer
move hgweb template code out to templater
r1896 common_filters = {
Vadim Gelfer
improve templating....
r1904 "addbreaks": nl2br,
Vadim Gelfer
make templater bit more flexible and efficient for external users.
r2191 "basename": os.path.basename,
Vadim Gelfer
move hgweb template code out to templater
r1896 "age": age,
"date": lambda x: util.datestr(x),
Vadim Gelfer
improve template errors when something is wrong.
r1905 "domain": domain,
Vadim Gelfer
add changelog style to command line template....
r1987 "email": email,
Vadim Gelfer
improve templating....
r1904 "escape": lambda x: cgi.escape(x, True),
Vadim Gelfer
add changelog style to command line template....
r1987 "fill68": lambda x: fill(x, width=68),
"fill76": lambda x: fill(x, width=76),
Brendan Cully
templater.firstline should handle empty strings
r2559 "firstline": firstline,
Vadim Gelfer
add changelog style to command line template....
r1987 "tabindent": lambda x: indent(x, '\t'),
Benoit Boissinot
add a new template function 'hgdate'...
r2519 "hgdate": hgdate,
Vadim Gelfer
add iso date template filter.
r1906 "isodate": isodate,
Vadim Gelfer
move hgweb template code out to templater
r1896 "obfuscate": obfuscate,
Vadim Gelfer
hide some functions behind lambdas, so demandload is useful.
r1912 "permissions": lambda x: x and "-rwxr-xr-x" or "-rw-r--r--",
Vadim Gelfer
improve templating....
r1904 "person": person,
"rfc822date": lambda x: util.datestr(x, "%a, %d %b %Y %H:%M:%S"),
Vadim Gelfer
hide some functions behind lambdas, so demandload is useful.
r1912 "short": lambda x: x[:12],
Vadim Gelfer
add changelog style to command line template....
r1987 "shortdate": shortdate,
"stringify": stringify,
Vadim Gelfer
improve templating....
r1904 "strip": lambda x: x.strip(),
Vadim Gelfer
hide some functions behind lambdas, so demandload is useful.
r1912 "urlescape": lambda x: urllib.quote(x),
"user": lambda x: util.shortuser(x),
Vadim Gelfer
move hgweb template code out to templater
r1896 }
Vadim Gelfer
move hgweb.templatepath into templater
r1897
Vadim Gelfer
many small changes to templater....
r1899 def templatepath(name=None):
Vadim Gelfer
add doc comments to template code.
r1909 '''return location of template file or directory (if no name).
returns None if not found.'''
Vadim Gelfer
move changeset_templater into templater module.
r2189
Lee Cantey
Windows py2exe version didn't handle names given to templatepath() correctly
r2001 # executable version (py2exe) doesn't support __file__
if hasattr(sys, 'frozen'):
module = sys.executable
else:
module = __file__
Vadim Gelfer
many small changes to templater....
r1899 for f in 'templates', '../templates':
fl = f.split('/')
if name: fl.append(name)
Lee Cantey
Windows py2exe version didn't handle names given to templatepath() correctly
r2001 p = os.path.join(os.path.dirname(module), *fl)
Vadim Gelfer
many small changes to templater....
r1899 if (name and os.path.exists(p)) or os.path.isdir(p):
Vadim Gelfer
move hgweb.templatepath into templater
r1897 return os.path.normpath(p)
Vadim Gelfer
move changeset_templater into templater module.
r2189
class changeset_templater(object):
'''format changeset information.'''
def __init__(self, ui, repo, mapfile, dest=None):
self.t = templater(mapfile, common_filters,
cache={'parent': '{rev}:{node|short} ',
'manifest': '{rev}:{node|short}'})
self.ui = ui
self.dest = dest
self.repo = repo
def use_template(self, t):
'''set template string to use'''
self.t.cache['changeset'] = t
def write(self, thing, header=False):
'''write expanded template.
uses in-order recursive traverse of iterators.'''
dest = self.dest or self.ui
for t in thing:
if hasattr(t, '__iter__'):
self.write(t, header=header)
elif header:
dest.write_header(t)
else:
dest.write(t)
def write_header(self, thing):
self.write(thing, header=True)
Vadim Gelfer
make templater bit more flexible and efficient for external users.
r2191 def show(self, rev=0, changenode=None, brinfo=None, changes=None,
**props):
Vadim Gelfer
move changeset_templater into templater module.
r2189 '''show a single changeset or file revision'''
log = self.repo.changelog
if changenode is None:
changenode = log.node(rev)
elif not rev:
rev = log.rev(changenode)
Vadim Gelfer
make templater bit more flexible and efficient for external users.
r2191 if changes is None:
changes = log.read(changenode)
Vadim Gelfer
move changeset_templater into templater module.
r2189
def showlist(name, values, plural=None, **args):
'''expand set of values.
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'.
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 space.
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'.
'''
if plural: names = plural
else: names = name + 's'
if not values:
noname = 'no_' + names
if noname in self.t:
yield self.t(noname, **args)
return
if name not in self.t:
if isinstance(values[0], str):
yield ' '.join(values)
else:
for v in values:
yield dict(v, **args)
return
startname = 'start_' + names
if startname in self.t:
yield self.t(startname, **args)
vargs = args.copy()
def one(v, tag=name):
try:
vargs.update(v)
except (AttributeError, ValueError):
try:
for a, b in v:
vargs[a] = b
except ValueError:
vargs[name] = v
return self.t(tag, **vargs)
lastname = 'last_' + name
if lastname in self.t:
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_' + names
if endname in self.t:
yield self.t(endname, **args)
if brinfo:
def showbranches(**args):
if changenode in brinfo:
for x in showlist('branch', brinfo[changenode],
plural='branches', **args):
yield x
else:
showbranches = ''
if self.ui.debugflag:
def showmanifest(**args):
args = args.copy()
args.update(dict(rev=self.repo.manifest.rev(changes[0]),
node=hex(changes[0])))
yield self.t('manifest', **args)
else:
showmanifest = ''
def showparents(**args):
parents = [[('rev', log.rev(p)), ('node', hex(p))]
for p in log.parents(changenode)
if self.ui.debugflag or p != nullid]
if (not self.ui.debugflag and len(parents) == 1 and
parents[0][0][1] == rev - 1):
return
for x in showlist('parent', parents, **args):
yield x
def showtags(**args):
for x in showlist('tag', self.repo.nodetags(changenode), **args):
yield x
if self.ui.debugflag:
files = self.repo.changes(log.parents(changenode)[0], changenode)
def showfiles(**args):
for x in showlist('file', files[0], **args): yield x
def showadds(**args):
for x in showlist('file_add', files[1], **args): yield x
def showdels(**args):
for x in showlist('file_del', files[2], **args): yield x
else:
def showfiles(**args):
for x in showlist('file', changes[3], **args): yield x
showadds = ''
showdels = ''
Vadim Gelfer
make templater bit more flexible and efficient for external users.
r2191 defprops = {
Vadim Gelfer
move changeset_templater into templater module.
r2189 'author': changes[1],
'branches': showbranches,
'date': changes[2],
'desc': changes[4],
'file_adds': showadds,
'file_dels': showdels,
'files': showfiles,
'manifest': showmanifest,
'node': hex(changenode),
'parents': showparents,
'rev': rev,
'tags': showtags,
}
Vadim Gelfer
make templater bit more flexible and efficient for external users.
r2191 props = props.copy()
props.update(defprops)
Vadim Gelfer
move changeset_templater into templater module.
r2189
try:
if self.ui.debugflag and 'header_debug' in self.t:
key = 'header_debug'
elif self.ui.quiet and 'header_quiet' in self.t:
key = 'header_quiet'
elif self.ui.verbose and 'header_verbose' in self.t:
key = 'header_verbose'
elif 'header' in self.t:
key = 'header'
else:
key = ''
if key:
self.write_header(self.t(key, **props))
if self.ui.debugflag and 'changeset_debug' in self.t:
key = 'changeset_debug'
elif self.ui.quiet and 'changeset_quiet' in self.t:
key = 'changeset_quiet'
elif self.ui.verbose and 'changeset_verbose' in self.t:
key = 'changeset_verbose'
else:
key = 'changeset'
self.write(self.t(key, **props))
except KeyError, inst:
raise util.Abort(_("%s: no key named '%s'") % (self.t.mapfile,
inst.args[0]))
except SyntaxError, inst:
raise util.Abort(_('%s: %s') % (self.t.mapfile, inst.args[0]))
Vadim Gelfer
move stringio class from bugzilla extension into templater module.
r2199
class stringio(object):
'''wrap cStringIO for use by changeset_templater.'''
def __init__(self):
self.fp = cStringIO.StringIO()
def write(self, *args):
for a in args:
self.fp.write(a)
write_header = write
Vadim Gelfer
make templater.stringio a tiny bit more use.
r2202 def __getattr__(self, key):
return getattr(self.fp, key)