1049
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
1 ##############################################################################
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
2 #
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
3 # Copyright (c) 2001 Zope Corporation and Contributors. All Rights Reserved.
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
4 #
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
5 # This software is subject to the provisions of the Zope Public License,
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
6 # Version 2.0 (ZPL). A copy of the ZPL should accompany this distribution.
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
7 # THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
8 # WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
9 # WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
10 # FOR A PARTICULAR PURPOSE
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
11 #
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
12 ##############################################################################
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
13 """TALES
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
14
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
15 An implementation of a generic TALES engine
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
16 """
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
17
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
18 __version__='$Revision: 1.1 $'[11:-2]
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
19
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
20 import re, sys
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
21 from roundup.cgi import ZTUtils
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
22 from MultiMapping import MultiMapping
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
23
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
24 StringType = type('')
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
25
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
26 NAME_RE = r"[a-zA-Z][a-zA-Z0-9_]*"
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
27 _parse_expr = re.compile(r"(%s):" % NAME_RE).match
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
28 _valid_name = re.compile('%s$' % NAME_RE).match
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
29
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
30 class TALESError(Exception):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
31 """Error during TALES expression evaluation"""
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
32
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
33 class Undefined(TALESError):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
34 '''Exception raised on traversal of an undefined path'''
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
35
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
36 class RegistrationError(Exception):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
37 '''TALES Type Registration Error'''
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
38
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
39 class CompilerError(Exception):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
40 '''TALES Compiler Error'''
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
41
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
42 class Default:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
43 '''Retain Default'''
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
44 Default = Default()
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
45
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
46 _marker = []
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
47
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
48 class SafeMapping(MultiMapping):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
49 '''Mapping with security declarations and limited method exposure.
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
50
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
51 Since it subclasses MultiMapping, this class can be used to wrap
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
52 one or more mapping objects. Restricted Python code will not be
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
53 able to mutate the SafeMapping or the wrapped mappings, but will be
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
54 able to read any value.
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
55 '''
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
56 __allow_access_to_unprotected_subobjects__ = 1
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
57 push = pop = None
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
58
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
59 _push = MultiMapping.push
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
60 _pop = MultiMapping.pop
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
61
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
62 def has_get(self, key, _marker=[]):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
63 v = self.get(key, _marker)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
64 return v is not _marker, v
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
65
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
66 class Iterator(ZTUtils.Iterator):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
67 def __init__(self, name, seq, context):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
68 ZTUtils.Iterator.__init__(self, seq)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
69 self.name = name
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
70 self._context = context
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
71
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
72 def next(self):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
73 if ZTUtils.Iterator.next(self):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
74 self._context.setLocal(self.name, self.item)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
75 return 1
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
76 return 0
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
77
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
78
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
79 class ErrorInfo:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
80 """Information about an exception passed to an on-error handler."""
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
81 __allow_access_to_unprotected_subobjects__ = 1
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
82
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
83 def __init__(self, err, position=(None, None)):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
84 if isinstance(err, Exception):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
85 self.type = err.__class__
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
86 self.value = err
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
87 else:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
88 self.type = err
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
89 self.value = None
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
90 self.lineno = position[0]
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
91 self.offset = position[1]
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
92
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
93
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
94 class Engine:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
95 '''Expression Engine
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
96
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
97 An instance of this class keeps a mutable collection of expression
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
98 type handlers. It can compile expression strings by delegating to
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
99 these handlers. It can provide an expression Context, which is
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
100 capable of holding state and evaluating compiled expressions.
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
101 '''
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
102 Iterator = Iterator
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
103
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
104 def __init__(self, Iterator=None):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
105 self.types = {}
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
106 if Iterator is not None:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
107 self.Iterator = Iterator
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
108
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
109 def registerType(self, name, handler):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
110 if not _valid_name(name):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
111 raise RegistrationError, 'Invalid Expression type "%s".' % name
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
112 types = self.types
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
113 if types.has_key(name):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
114 raise RegistrationError, (
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
115 'Multiple registrations for Expression type "%s".' %
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
116 name)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
117 types[name] = handler
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
118
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
119 def getTypes(self):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
120 return self.types
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
121
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
122 def compile(self, expression):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
123 m = _parse_expr(expression)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
124 if m:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
125 type = m.group(1)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
126 expr = expression[m.end():]
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
127 else:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
128 type = "standard"
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
129 expr = expression
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
130 try:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
131 handler = self.types[type]
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
132 except KeyError:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
133 raise CompilerError, (
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
134 'Unrecognized expression type "%s".' % type)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
135 return handler(type, expr, self)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
136
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
137 def getContext(self, contexts=None, **kwcontexts):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
138 if contexts is not None:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
139 if kwcontexts:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
140 kwcontexts.update(contexts)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
141 else:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
142 kwcontexts = contexts
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
143 return Context(self, kwcontexts)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
144
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
145 def getCompilerError(self):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
146 return CompilerError
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
147
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
148 class Context:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
149 '''Expression Context
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
150
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
151 An instance of this class holds context information that it can
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
152 use to evaluate compiled expressions.
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
153 '''
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
154
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
155 _context_class = SafeMapping
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
156 position = (None, None)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
157 source_file = None
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
158
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
159 def __init__(self, engine, contexts):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
160 self._engine = engine
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
161 self.contexts = contexts
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
162 contexts['nothing'] = None
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
163 contexts['default'] = Default
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
164
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
165 self.repeat_vars = rv = {}
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
166 # Wrap this, as it is visible to restricted code
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
167 contexts['repeat'] = rep = self._context_class(rv)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
168 contexts['loop'] = rep # alias
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
169
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
170 self.global_vars = gv = contexts.copy()
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
171 self.local_vars = lv = {}
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
172 self.vars = self._context_class(gv, lv)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
173
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
174 # Keep track of what needs to be popped as each scope ends.
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
175 self._scope_stack = []
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
176
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
177 def beginScope(self):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
178 self._scope_stack.append([self.local_vars.copy()])
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
179
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
180 def endScope(self):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
181 scope = self._scope_stack.pop()
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
182 self.local_vars = lv = scope[0]
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
183 v = self.vars
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
184 v._pop()
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
185 v._push(lv)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
186 # Pop repeat variables, if any
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
187 i = len(scope) - 1
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
188 while i:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
189 name, value = scope[i]
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
190 if value is None:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
191 del self.repeat_vars[name]
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
192 else:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
193 self.repeat_vars[name] = value
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
194 i = i - 1
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
195
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
196 def setLocal(self, name, value):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
197 self.local_vars[name] = value
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
198
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
199 def setGlobal(self, name, value):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
200 self.global_vars[name] = value
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
201
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
202 def setRepeat(self, name, expr):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
203 expr = self.evaluate(expr)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
204 if not expr:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
205 return self._engine.Iterator(name, (), self)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
206 it = self._engine.Iterator(name, expr, self)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
207 old_value = self.repeat_vars.get(name)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
208 self._scope_stack[-1].append((name, old_value))
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
209 self.repeat_vars[name] = it
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
210 return it
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
211
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
212 def evaluate(self, expression,
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
213 isinstance=isinstance, StringType=StringType):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
214 if isinstance(expression, StringType):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
215 expression = self._engine.compile(expression)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
216 __traceback_supplement__ = (
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
217 TALESTracebackSupplement, self, expression)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
218 v = expression(self)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
219 return v
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
220
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
221 evaluateValue = evaluate
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
222
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
223 def evaluateBoolean(self, expr):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
224 return not not self.evaluate(expr)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
225
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
226 def evaluateText(self, expr, None=None):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
227 text = self.evaluate(expr)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
228 if text is Default or text is None:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
229 return text
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
230 return str(text)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
231
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
232 def evaluateStructure(self, expr):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
233 return self.evaluate(expr)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
234 evaluateStructure = evaluate
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
235
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
236 def evaluateMacro(self, expr):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
237 # XXX Should return None or a macro definition
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
238 return self.evaluate(expr)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
239 evaluateMacro = evaluate
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
240
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
241 def createErrorInfo(self, err, position):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
242 return ErrorInfo(err, position)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
243
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
244 def getDefault(self):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
245 return Default
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
246
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
247 def setSourceFile(self, source_file):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
248 self.source_file = source_file
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
249
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
250 def setPosition(self, position):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
251 self.position = position
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
252
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
253
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
254
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
255 class TALESTracebackSupplement:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
256 """Implementation of ITracebackSupplement"""
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
257 def __init__(self, context, expression):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
258 self.context = context
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
259 self.source_url = context.source_file
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
260 self.line = context.position[0]
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
261 self.column = context.position[1]
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
262 self.expression = repr(expression)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
263
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
264 def getInfo(self, as_html=0):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
265 import pprint
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
266 data = self.context.contexts.copy()
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
267 s = pprint.pformat(data)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
268 if not as_html:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
269 return ' - Names:\n %s' % string.replace(s, '\n', '\n ')
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
270 else:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
271 from cgi import escape
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
272 return '<b>Names:</b><pre>%s</pre>' % (escape(s))
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
273 return None
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
274
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
275
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
276
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
277 class SimpleExpr:
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
278 '''Simple example of an expression type handler'''
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
279 def __init__(self, name, expr, engine):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
280 self._name = name
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
281 self._expr = expr
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
282 def __call__(self, econtext):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
283 return self._name, self._expr
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
284 def __repr__(self):
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
285 return '<SimpleExpr %s %s>' % (self._name, `self._expr`)
|
Richard Jones <richard@users.sourceforge.net>
parents:
diff
changeset
|
286
|