##// END OF EJS Templates
Merge with stable
Merge with stable

File last commit:

r4335:f4a1eac5 merge default
r4335:f4a1eac5 merge default
Show More
templater.py
290 lines | 9.0 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.
Matt Mackall
Simplify i18n imports
r3891 from i18n import _
Vadim Gelfer
move changeset_templater into templater module.
r2189 from node import *
Matt Mackall
Replace demandload with new demandimport
r3877 import cgi, re, sys, os, time, urllib, util, textwrap
Vadim Gelfer
use safer string parser for template engine.
r1901
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
make parsestring work with strings that do not have quotes.
r1902 if quoted:
Matt Mackall
templater: simplify parsestring
r3639 if len(s) < 2 or s[0] != s[-1]:
raise SyntaxError(_('unmatched quotes'))
return s[1:-1].decode('string_escape')
Matt Mackall
templater: use str.decode in parse_string
r3632
return s.decode('string_escape')
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
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|...}.'''
Matt Mackall
template: fold template() into __call__, minor optimizations...
r3638 template_re = re.compile(r"(?:(?:#(?=[\w\|%]+#))|(?:{(?=[\w\|%]+})))"
r"(\w+)(?:(?:%(\w+))|((?:\|\w+)*))[#}]")
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):
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
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.'''
Matt Mackall
templater: simplify cache and remove filter argument in __call__
r3637 if not self.cache.has_key(t):
Vadim Gelfer
improve template errors when something is wrong.
r1905 try:
Matt Mackall
templater: simplify cache and remove filter argument in __call__
r3637 self.cache[t] = file(self.map[t]).read()
Vadim Gelfer
improve template errors when something is wrong.
r1905 except IOError, inst:
raise IOError(inst.args[0], _('template file %s: %s') %
(self.map[t], inst.args[1]))
Matt Mackall
template: fold template() into __call__, minor optimizations...
r3638 tmpl = self.cache[t]
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)
Matt Mackall
template: fold template() into __call__, minor optimizations...
r3638 if not m:
yield tmpl
break
Vadim Gelfer
move hgweb template code out to templater
r1896
Matt Mackall
template: fold template() into __call__, minor optimizations...
r3638 start, end = m.span(0)
key, format, fl = m.groups()
Matt Mackall
templater: simplify template function
r3636
Matt Mackall
template: fold template() into __call__, minor optimizations...
r3638 if start:
yield tmpl[:start]
tmpl = tmpl[end:]
Matt Mackall
templater: simplify template function
r3636
Matt Mackall
template: fold template() into __call__, minor optimizations...
r3638 if key in map:
v = map[key]
else:
v = self.defaults.get(key, "")
if callable(v):
v = v(**map)
if format:
if not hasattr(v, '__iter__'):
raise SyntaxError(_("Error expanding '%s%s'")
% (key, format))
lm = map.copy()
for i in v:
lm.update(i)
yield self(format, **lm)
else:
if fl:
Vadim Gelfer
move hgweb template code out to templater
r1896 for f in fl.split("|")[1:]:
Matt Mackall
templater: simplify cache and remove filter argument in __call__
r3637 v = self.filters[f](v)
Vadim Gelfer
move hgweb template code out to templater
r1896 yield v
Vadim Gelfer
move repeated work out of inner loops.
r1900
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.'''
Matt Mackall
templater: speed up changeset writes and stringify
r3641 if hasattr(thing, '__iter__'):
Matt Mackall
further simplify stringify
r3647 return "".join([stringify(t) for t in thing if t is not None])
Matt Mackall
templater: speed up changeset writes and stringify
r3641 return str(thing)
Vadim Gelfer
add changelog style to command line template....
r1987
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
Matt Mackall
templater: remove cStringIO for fill
r3633 return "".join([space_re.sub(' ', textwrap.fill(para, width)) + rest
for para, rest in findparas()])
Vadim Gelfer
add changelog style to command line template....
r1987
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):
Alexis S. L. Carvalho
templater.py: fix obfuscate
r3904 text = unicode(text, util._encoding, '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.'''
lines = text.splitlines()
num_lines = len(lines)
Matt Mackall
templater: take cStringIO out of indent
r3635 def indenter():
for i in xrange(num_lines):
l = lines[i]
if i and l.strip():
yield prefix
yield l
if i < num_lines - 1 or text.endswith('\n'):
yield '\n'
return "".join(indenter())
Vadim Gelfer
add changelog style to command line template....
r1987
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),
Thomas Arendsen Hein
Added extra changeset info to templater and map-cmdline.default.
r3462 "stringescape": lambda x: x.encode('string_escape'),
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