annotate genshi/template/interpolation.py @ 500:0742f421caba experimental-inline

Merged revisions 487-603 via svnmerge from http://svn.edgewall.org/repos/genshi/trunk
author cmlenz
date Fri, 01 Jun 2007 17:21:47 +0000
parents
children 1837f39efd6f
rev   line source
500
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
1 # -*- coding: utf-8 -*-
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
2 #
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
3 # Copyright (C) 2007 Edgewall Software
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
4 # All rights reserved.
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
5 #
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
6 # This software is licensed as described in the file COPYING, which
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
7 # you should have received as part of this distribution. The terms
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
8 # are also available at http://genshi.edgewall.org/wiki/License.
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
9 #
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
10 # This software consists of voluntary contributions made by many
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
11 # individuals. For the exact contribution history, see the revision
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
12 # history and logs, available at http://genshi.edgewall.org/log/.
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
13
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
14 """String interpolation routines, i.e. the splitting up a given text into some
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
15 parts that are literal strings, and others that are Python expressions.
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
16 """
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
17
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
18 from itertools import chain
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
19 import os
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
20 from tokenize import tokenprog
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
21
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
22 from genshi.core import TEXT
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
23 from genshi.template.base import TemplateSyntaxError, EXPR
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
24 from genshi.template.eval import Expression
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
25
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
26 __all__ = ['interpolate']
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
27 __docformat__ = 'restructuredtext en'
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
28
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
29 NAMESTART = 'abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ_'
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
30 NAMECHARS = NAMESTART + '.0123456789'
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
31 PREFIX = '$'
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
32
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
33 def interpolate(text, basedir=None, filename=None, lineno=-1, offset=0,
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
34 lookup='lenient'):
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
35 """Parse the given string and extract expressions.
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
36
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
37 This function is a generator that yields `TEXT` events for literal strings,
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
38 and `EXPR` events for expressions, depending on the results of parsing the
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
39 string.
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
40
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
41 >>> for kind, data, pos in interpolate("hey ${foo}bar"):
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
42 ... print kind, `data`
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
43 TEXT u'hey '
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
44 EXPR Expression('foo')
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
45 TEXT u'bar'
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
46
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
47 :param text: the text to parse
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
48 :param basedir: base directory of the file in which the text was found
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
49 (optional)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
50 :param filename: basename of the file in which the text was found (optional)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
51 :param lineno: the line number at which the text was found (optional)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
52 :param offset: the column number at which the text starts in the source
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
53 (optional)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
54 :param lookup: the variable lookup mechanism; either "lenient" (the
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
55 default), "strict", or a custom lookup class
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
56 :return: a list of `TEXT` and `EXPR` events
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
57 :raise TemplateSyntaxError: when a syntax error in an expression is
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
58 encountered
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
59 """
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
60 filepath = filename
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
61 if filepath and basedir:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
62 filepath = os.path.join(basedir, filepath)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
63 pos = [filepath, lineno, offset]
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
64
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
65 textbuf = []
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
66 textpos = None
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
67 for is_expr, chunk in chain(lex(text, pos, filepath), [(True, '')]):
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
68 if is_expr:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
69 if textbuf:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
70 yield TEXT, u''.join(textbuf), textpos
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
71 del textbuf[:]
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
72 textpos = None
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
73 if chunk:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
74 try:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
75 expr = Expression(chunk.strip(), pos[0], pos[1],
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
76 lookup=lookup)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
77 yield EXPR, expr, tuple(pos)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
78 except SyntaxError, err:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
79 raise TemplateSyntaxError(err, filepath, pos[1],
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
80 pos[2] + (err.offset or 0))
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
81 else:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
82 textbuf.append(chunk)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
83 if textpos is None:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
84 textpos = tuple(pos)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
85
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
86 if '\n' in chunk:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
87 lines = chunk.splitlines()
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
88 pos[1] += len(lines) - 1
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
89 pos[2] += len(lines[-1])
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
90 else:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
91 pos[2] += len(chunk)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
92
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
93 def lex(text, textpos, filepath):
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
94 offset = pos = 0
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
95 end = len(text)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
96 escaped = False
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
97
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
98 while 1:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
99 if escaped:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
100 offset = text.find(PREFIX, offset + 2)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
101 escaped = False
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
102 else:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
103 offset = text.find(PREFIX, pos)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
104 if offset < 0 or offset == end - 1:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
105 break
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
106 next = text[offset + 1]
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
107
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
108 if next == '{':
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
109 if offset > pos:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
110 yield False, text[pos:offset]
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
111 pos = offset + 2
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
112 level = 1
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
113 while level:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
114 match = tokenprog.match(text, pos)
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
115 if match is None:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
116 raise TemplateSyntaxError('invalid syntax', filepath,
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
117 *textpos[1:])
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
118 pos = match.end()
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
119 tstart, tend = match.regs[3]
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
120 token = text[tstart:tend]
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
121 if token == '{':
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
122 level += 1
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
123 elif token == '}':
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
124 level -= 1
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
125 yield True, text[offset + 2:pos - 1]
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
126
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
127 elif next in NAMESTART:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
128 if offset > pos:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
129 yield False, text[pos:offset]
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
130 pos = offset
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
131 pos += 1
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
132 while pos < end:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
133 char = text[pos]
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
134 if char not in NAMECHARS:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
135 break
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
136 pos += 1
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
137 yield True, text[offset + 1:pos].strip()
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
138
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
139 elif not escaped and next == PREFIX:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
140 escaped = True
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
141 pos = offset + 1
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
142
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
143 else:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
144 yield False, text[pos:offset + 1]
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
145 pos = offset + 1
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
146
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
147 if pos < end:
0742f421caba Merged revisions 487-603 via svnmerge from
cmlenz
parents:
diff changeset
148 yield False, text[pos:]
Copyright (C) 2012-2017 Edgewall Software