142
142
# PyInstaller environment doesn't include this module.
143
143
DISALLOW_FUNCTIONS .add (help )
144
144
145
+ # Opt-in type safety experiment. Will be opt-out in 2.x
146
+
147
+ BASIC_ALLOWED_ATTRS = {
148
+ int : {
149
+ "as_integer_ratio" ,
150
+ "bit_length" ,
151
+ "conjugate" ,
152
+ "denominator" ,
153
+ "from_bytes" ,
154
+ "imag" ,
155
+ "numerator" ,
156
+ "real" ,
157
+ "to_bytes" ,
158
+ },
159
+ float : {
160
+ "as_integer_ratio" ,
161
+ "conjugate" ,
162
+ "fromhex" ,
163
+ "hex" ,
164
+ "imag" ,
165
+ "is_integer" ,
166
+ "real" ,
167
+ },
168
+ str : {
169
+ "capitalize" ,
170
+ "casefold" ,
171
+ "center" ,
172
+ "count" ,
173
+ "encode" ,
174
+ "endswith" ,
175
+ "expandtabs" ,
176
+ "find" ,
177
+ "format" ,
178
+ "format_map" ,
179
+ "index" ,
180
+ "isalnum" ,
181
+ "isalpha" ,
182
+ "isascii" ,
183
+ "isdecimal" ,
184
+ "isdigit" ,
185
+ "isidentifier" ,
186
+ "islower" ,
187
+ "isnumeric" ,
188
+ "isprintable" ,
189
+ "isspace" ,
190
+ "istitle" ,
191
+ "isupper" ,
192
+ "join" ,
193
+ "ljust" ,
194
+ "lower" ,
195
+ "lstrip" ,
196
+ "maketrans" ,
197
+ "partition" ,
198
+ "removeprefix" ,
199
+ "removesuffix" ,
200
+ "replace" ,
201
+ "rfind" ,
202
+ "rindex" ,
203
+ "rjust" ,
204
+ "rpartition" ,
205
+ "rsplit" ,
206
+ "rstrip" ,
207
+ "split" ,
208
+ "splitlines" ,
209
+ "startswith" ,
210
+ "strip" ,
211
+ "swapcase" ,
212
+ "title" ,
213
+ "translate" ,
214
+ "upper" ,
215
+ "zfill" ,
216
+ },
217
+ bool : {
218
+ "as_integer_ratio" ,
219
+ "bit_length" ,
220
+ "conjugate" ,
221
+ "denominator" ,
222
+ "from_bytes" ,
223
+ "imag" ,
224
+ "numerator" ,
225
+ "real" ,
226
+ "to_bytes" ,
227
+ },
228
+ None : {},
229
+ dict : {
230
+ "clear" ,
231
+ "copy" ,
232
+ "fromkeys" ,
233
+ "get" ,
234
+ "items" ,
235
+ "keys" ,
236
+ "pop" ,
237
+ "popitem" ,
238
+ "setdefault" ,
239
+ "update" ,
240
+ "values" ,
241
+ },
242
+ list : {
243
+ "pop" ,
244
+ "append" ,
245
+ "index" ,
246
+ "reverse" ,
247
+ "count" ,
248
+ "sort" ,
249
+ "copy" ,
250
+ "extend" ,
251
+ "clear" ,
252
+ "insert" ,
253
+ "remove" ,
254
+ },
255
+ set : {
256
+ "pop" ,
257
+ "intersection_update" ,
258
+ "intersection" ,
259
+ "issubset" ,
260
+ "symmetric_difference_update" ,
261
+ "discard" ,
262
+ "isdisjoint" ,
263
+ "difference_update" ,
264
+ "issuperset" ,
265
+ "add" ,
266
+ "copy" ,
267
+ "union" ,
268
+ "clear" ,
269
+ "update" ,
270
+ "symmetric_difference" ,
271
+ "difference" ,
272
+ "remove" ,
273
+ },
274
+ tuple : {"index" , "count" },
275
+ }
276
+
145
277
146
278
########################################
147
279
# Exceptions:
148
280
149
281
282
+ class TypeNotSpecified (Exception ):
283
+ pass
284
+
285
+
150
286
class InvalidExpression (Exception ):
151
287
"""Generic Exception"""
152
288
@@ -344,7 +480,7 @@ class SimpleEval(object): # pylint: disable=too-few-public-methods
344
480
345
481
expr = ""
346
482
347
- def __init__ (self , operators = None , functions = None , names = None ):
483
+ def __init__ (self , operators = None , functions = None , names = None , allowed_attrs = None ):
348
484
"""
349
485
Create the evaluator instance. Set up valid operators (+,-, etc)
350
486
functions (add, random, get_val, whatever) and names."""
@@ -359,6 +495,7 @@ def __init__(self, operators=None, functions=None, names=None):
359
495
self .operators = operators
360
496
self .functions = functions
361
497
self .names = names
498
+ self .allowed_attrs = allowed_attrs
362
499
363
500
self .nodes = {
364
501
ast .Expr : self ._eval_expr ,
@@ -587,6 +724,8 @@ def _eval_subscript(self, node):
587
724
return container [key ]
588
725
589
726
def _eval_attribute (self , node ):
727
+ # DISALLOW_PREFIXES & DISALLOW_METHODS are global, there's never any access to
728
+ # attrs with these names, so we can bail early:
590
729
for prefix in DISALLOW_PREFIXES :
591
730
if node .attr .startswith (prefix ):
592
731
raise FeatureNotAvailable (
@@ -598,9 +737,27 @@ def _eval_attribute(self, node):
598
737
raise FeatureNotAvailable (
599
738
"Sorry, this method is not available. " "({0})" .format (node .attr )
600
739
)
601
- # eval node
740
+
741
+ # Evaluate "node" - the thing that we're trying to access an attr of first:
602
742
node_evaluated = self ._eval (node .value )
603
743
744
+ # If we've opted in to the 'allowed_attrs' checking per type, then since we now
745
+ # know what kind of node we've got, we can check if we're permitted to access this
746
+ # attr name on this node:
747
+ if self .allowed_attrs is not None :
748
+ type_to_check = type (node_evaluated )
749
+
750
+ allowed_attrs = self .allowed_attrs .get (type_to_check , TypeNotSpecified )
751
+ if allowed_attrs == TypeNotSpecified :
752
+ raise FeatureNotAvailable (
753
+ f"Sorry, attribute access not allowed on '{ type_to_check } '"
754
+ f" (attempted to access `.{ node .attr } `)"
755
+ )
756
+ if node .attr not in allowed_attrs :
757
+ raise FeatureNotAvailable (
758
+ f"Sorry, '.{ node .attr } ' access not allowed on '{ type_to_check } '"
759
+ )
760
+
604
761
# Maybe the base object is an actual object, not just a dict
605
762
try :
606
763
return getattr (node_evaluated , node .attr )
@@ -762,7 +919,12 @@ def do_generator(gi=0):
762
919
return to_return
763
920
764
921
765
- def simple_eval (expr , operators = None , functions = None , names = None ):
922
+ def simple_eval (expr , operators = None , functions = None , names = None , allowed_attrs = None ):
766
923
"""Simply evaluate an expresssion"""
767
- s = SimpleEval (operators = operators , functions = functions , names = names )
924
+ s = SimpleEval (
925
+ operators = operators ,
926
+ functions = functions ,
927
+ names = names ,
928
+ allowed_attrs = allowed_attrs ,
929
+ )
768
930
return s .eval (expr )
0 commit comments