annotate roundup/cgi/PageTemplates/Expressions.py @ 1142:eac669d738d1 0.5.0-b2

doc
author Richard Jones <richard@users.sourceforge.net>
date Fri, 13 Sep 2002 04:57:38 +0000
parents 7e193bbda38e
children 49aebf5a8691
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
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
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
14 """Page Template Expression Engine
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
15
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
16 Page Template-specific implementation of TALES, with handlers
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
17 for Python expressions, string literals, and paths.
1071
c08b3820edd1 Adhering to ZPL
Richard Jones <richard@users.sourceforge.net>
parents: 1065
diff changeset
18
c08b3820edd1 Adhering to ZPL
Richard Jones <richard@users.sourceforge.net>
parents: 1065
diff changeset
19
c08b3820edd1 Adhering to ZPL
Richard Jones <richard@users.sourceforge.net>
parents: 1065
diff changeset
20 Modified for Roundup 0.5 release:
c08b3820edd1 Adhering to ZPL
Richard Jones <richard@users.sourceforge.net>
parents: 1065
diff changeset
21
c08b3820edd1 Adhering to ZPL
Richard Jones <richard@users.sourceforge.net>
parents: 1065
diff changeset
22 - Removed all Zope-specific code (doesn't even try to import that stuff now)
c08b3820edd1 Adhering to ZPL
Richard Jones <richard@users.sourceforge.net>
parents: 1065
diff changeset
23 - Removed all Acquisition
c08b3820edd1 Adhering to ZPL
Richard Jones <richard@users.sourceforge.net>
parents: 1065
diff changeset
24 - Made traceback info more informative
c08b3820edd1 Adhering to ZPL
Richard Jones <richard@users.sourceforge.net>
parents: 1065
diff changeset
25
1049
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
26 """
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
27
1136
7e193bbda38e added generic item editing
Richard Jones <richard@users.sourceforge.net>
parents: 1101
diff changeset
28 __version__='$Revision: 1.6 $'[11:-2]
1049
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
29
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
30 import re, sys
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
31 from TALES import Engine, CompilerError, _valid_name, NAME_RE, \
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
32 Undefined, Default, _parse_expr
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
33 from string import strip, split, join, replace, lstrip
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
34
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
35 _engine = None
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
36 def getEngine():
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
37 global _engine
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
38 if _engine is None:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
39 from PathIterator import Iterator
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
40 _engine = Engine(Iterator)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
41 installHandlers(_engine)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
42 return _engine
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
43
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
44 def installHandlers(engine):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
45 reg = engine.registerType
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
46 pe = PathExpr
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
47 for pt in ('standard', 'path', 'exists', 'nocall'):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
48 reg(pt, pe)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
49 reg('string', StringExpr)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
50 reg('python', PythonExpr)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
51 reg('not', NotExpr)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
52 reg('defer', DeferExpr)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
53
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
54 from PythonExpr import getSecurityManager, PythonExpr
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
55 try:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
56 from zExceptions import Unauthorized
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
57 except ImportError:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
58 Unauthorized = "Unauthorized"
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
59 def call_with_ns(f, ns, arg=1):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
60 if arg==2:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
61 return f(None, ns)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
62 else:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
63 return f(ns)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
64
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
65 class _SecureModuleImporter:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
66 """Simple version of the importer for use with trusted code."""
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
67 __allow_access_to_unprotected_subobjects__ = 1
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
68 def __getitem__(self, module):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
69 __import__(module)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
70 return sys.modules[module]
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
71
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
72 Undefs = (Undefined, AttributeError, KeyError,
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
73 TypeError, IndexError, Unauthorized)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
74
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
75 def render(ob, ns):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
76 """
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
77 Calls the object, possibly a document template, or just returns it if
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
78 not callable. (From DT_Util.py)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
79 """
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
80 if hasattr(ob, '__render_with_namespace__'):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
81 ob = call_with_ns(ob.__render_with_namespace__, ns)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
82 else:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
83 base = ob
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
84 if callable(base):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
85 try:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
86 if getattr(base, 'isDocTemp', 0):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
87 ob = call_with_ns(ob, ns, 2)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
88 else:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
89 ob = ob()
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
90 except AttributeError, n:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
91 if str(n) != '__call__':
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
92 raise
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
93 return ob
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
94
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
95 class SubPathExpr:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
96 def __init__(self, path):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
97 self._path = path = split(strip(path), '/')
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
98 self._base = base = path.pop(0)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
99 if not _valid_name(base):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
100 raise CompilerError, 'Invalid variable name "%s"' % base
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
101 # Parse path
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
102 self._dp = dp = []
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
103 for i in range(len(path)):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
104 e = path[i]
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
105 if e[:1] == '?' and _valid_name(e[1:]):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
106 dp.append((i, e[1:]))
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
107 dp.reverse()
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
108
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
109 def _eval(self, econtext,
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
110 list=list, isinstance=isinstance, StringType=type('')):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
111 vars = econtext.vars
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
112 path = self._path
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
113 if self._dp:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
114 path = list(path) # Copy!
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
115 for i, varname in self._dp:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
116 val = vars[varname]
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
117 if isinstance(val, StringType):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
118 path[i] = val
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
119 else:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
120 # If the value isn't a string, assume it's a sequence
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
121 # of path names.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
122 path[i:i+1] = list(val)
1065
0f9aa62917bd much nicer error messages when there's a templating error
Richard Jones <richard@users.sourceforge.net>
parents: 1049
diff changeset
123 base = self._base
1136
7e193bbda38e added generic item editing
Richard Jones <richard@users.sourceforge.net>
parents: 1101
diff changeset
124 __traceback_info__ = 'path expression "%s"'%('/'.join(self._path))
1049
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
125 if base == 'CONTEXTS':
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
126 ob = econtext.contexts
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
127 else:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
128 ob = vars[base]
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
129 if isinstance(ob, DeferWrapper):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
130 ob = ob()
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
131 if path:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
132 ob = restrictedTraverse(ob, path, getSecurityManager())
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
133 return ob
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
134
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
135 class PathExpr:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
136 def __init__(self, name, expr, engine):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
137 self._s = expr
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
138 self._name = name
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
139 paths = split(expr, '|')
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
140 self._subexprs = []
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
141 add = self._subexprs.append
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
142 for i in range(len(paths)):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
143 path = lstrip(paths[i])
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
144 if _parse_expr(path):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
145 # This part is the start of another expression type,
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
146 # so glue it back together and compile it.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
147 add(engine.compile(lstrip(join(paths[i:], '|'))))
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
148 break
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
149 add(SubPathExpr(path)._eval)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
150
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
151 def _exists(self, econtext):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
152 for expr in self._subexprs:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
153 try:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
154 expr(econtext)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
155 except Undefs:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
156 pass
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
157 else:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
158 return 1
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
159 return 0
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
160
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
161 def _eval(self, econtext,
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
162 isinstance=isinstance, StringType=type(''), render=render):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
163 for expr in self._subexprs[:-1]:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
164 # Try all but the last subexpression, skipping undefined ones.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
165 try:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
166 ob = expr(econtext)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
167 except Undefs:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
168 pass
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
169 else:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
170 break
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
171 else:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
172 # On the last subexpression allow exceptions through.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
173 ob = self._subexprs[-1](econtext)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
174
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
175 if self._name == 'nocall' or isinstance(ob, StringType):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
176 return ob
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
177 # Return the rendered object
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
178 return render(ob, econtext.vars)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
179
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
180 def __call__(self, econtext):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
181 if self._name == 'exists':
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
182 return self._exists(econtext)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
183 return self._eval(econtext)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
184
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
185 def __str__(self):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
186 return '%s expression %s' % (self._name, `self._s`)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
187
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
188 def __repr__(self):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
189 return '%s:%s' % (self._name, `self._s`)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
190
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
191
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
192 _interp = re.compile(r'\$(%(n)s)|\${(%(n)s(?:/%(n)s)*)}' % {'n': NAME_RE})
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
193
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
194 class StringExpr:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
195 def __init__(self, name, expr, engine):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
196 self._s = expr
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
197 if '%' in expr:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
198 expr = replace(expr, '%', '%%')
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
199 self._vars = vars = []
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
200 if '$' in expr:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
201 parts = []
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
202 for exp in split(expr, '$$'):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
203 if parts: parts.append('$')
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
204 m = _interp.search(exp)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
205 while m is not None:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
206 parts.append(exp[:m.start()])
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
207 parts.append('%s')
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
208 vars.append(PathExpr('path', m.group(1) or m.group(2),
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
209 engine))
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
210 exp = exp[m.end():]
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
211 m = _interp.search(exp)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
212 if '$' in exp:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
213 raise CompilerError, (
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
214 '$ must be doubled or followed by a simple path')
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
215 parts.append(exp)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
216 expr = join(parts, '')
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
217 self._expr = expr
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
218
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
219 def __call__(self, econtext):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
220 vvals = []
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
221 for var in self._vars:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
222 v = var(econtext)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
223 if isinstance(v, Exception):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
224 raise v
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
225 vvals.append(v)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
226 return self._expr % tuple(vvals)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
227
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
228 def __str__(self):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
229 return 'string expression %s' % `self._s`
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
230
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
231 def __repr__(self):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
232 return 'string:%s' % `self._s`
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
233
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
234 class NotExpr:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
235 def __init__(self, name, expr, compiler):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
236 self._s = expr = lstrip(expr)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
237 self._c = compiler.compile(expr)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
238
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
239 def __call__(self, econtext):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
240 return not econtext.evaluateBoolean(self._c)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
241
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
242 def __repr__(self):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
243 return 'not:%s' % `self._s`
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
244
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
245 class DeferWrapper:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
246 def __init__(self, expr, econtext):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
247 self._expr = expr
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
248 self._econtext = econtext
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
249
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
250 def __str__(self):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
251 return str(self())
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
252
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
253 def __call__(self):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
254 return self._expr(self._econtext)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
255
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
256 class DeferExpr:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
257 def __init__(self, name, expr, compiler):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
258 self._s = expr = lstrip(expr)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
259 self._c = compiler.compile(expr)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
260
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
261 def __call__(self, econtext):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
262 return DeferWrapper(self._c, econtext)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
263
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
264 def __repr__(self):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
265 return 'defer:%s' % `self._s`
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
266
1136
7e193bbda38e added generic item editing
Richard Jones <richard@users.sourceforge.net>
parents: 1101
diff changeset
267 class TraversalError:
7e193bbda38e added generic item editing
Richard Jones <richard@users.sourceforge.net>
parents: 1101
diff changeset
268 def __init__(self, path, name):
7e193bbda38e added generic item editing
Richard Jones <richard@users.sourceforge.net>
parents: 1101
diff changeset
269 self.path = path
7e193bbda38e added generic item editing
Richard Jones <richard@users.sourceforge.net>
parents: 1101
diff changeset
270 self.name = name
1049
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
271
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
272 def restrictedTraverse(self, path, securityManager,
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
273 get=getattr, has=hasattr, N=None, M=[],
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
274 TupleType=type(()) ):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
275
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
276 REQUEST = {'path': path}
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
277 REQUEST['TraversalRequestNameStack'] = path = path[:] # Copy!
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
278 if not path[0]:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
279 # If the path starts with an empty string, go to the root first.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
280 self = self.getPhysicalRoot()
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
281 path.pop(0)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
282
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
283 path.reverse()
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
284 object = self
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
285 #print 'TRAVERSE', (object, path)
1101
1ce96bf6482f better traversal error info
Richard Jones <richard@users.sourceforge.net>
parents: 1079
diff changeset
286 done = []
1049
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
287 while path:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
288 name = path.pop()
1136
7e193bbda38e added generic item editing
Richard Jones <richard@users.sourceforge.net>
parents: 1101
diff changeset
289 __traceback_info__ = TraversalError(done, name)
1049
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
290
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
291 if isinstance(name, TupleType):
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
292 object = apply(object, name)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
293 continue
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
294
1079
04b44df7af1d removed some property name clashes
Richard Jones <richard@users.sourceforge.net>
parents: 1071
diff changeset
295 # if name[0] == '_':
04b44df7af1d removed some property name clashes
Richard Jones <richard@users.sourceforge.net>
parents: 1071
diff changeset
296 # # Never allowed in a URL.
04b44df7af1d removed some property name clashes
Richard Jones <richard@users.sourceforge.net>
parents: 1071
diff changeset
297 # raise AttributeError, name
1049
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
298
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
299 # Try an attribute.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
300 o = get(object, name, M)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
301 # print '...', (object, name, M, o)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
302 if o is M:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
303 # Try an item.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
304 # print '... try an item'
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
305 try:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
306 # XXX maybe in Python 2.2 we can just check whether
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
307 # the object has the attribute "__getitem__"
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
308 # instead of blindly catching exceptions.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
309 o = object[name]
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
310 except AttributeError, exc:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
311 if str(exc).find('__getitem__') >= 0:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
312 # The object does not support the item interface.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
313 # Try to re-raise the original attribute error.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
314 # XXX I think this only happens with
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
315 # ExtensionClass instances.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
316 get(object, name)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
317 raise
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
318 except TypeError, exc:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
319 if str(exc).find('unsubscriptable') >= 0:
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
320 # The object does not support the item interface.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
321 # Try to re-raise the original attribute error.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
322 # XXX This is sooooo ugly.
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
323 get(object, name)
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
324 raise
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
325 #print '... object is now', `o`
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
326 object = o
1136
7e193bbda38e added generic item editing
Richard Jones <richard@users.sourceforge.net>
parents: 1101
diff changeset
327 done.append((name, o))
1049
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
328
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
329 return object
Richard Jones <richard@users.sourceforge.net>
parents:
diff changeset
330

Roundup Issue Tracker: http://roundup-tracker.org/