############################################################################## # # Copyright (c) 2003 Zope Corporation and Contributors. # All Rights Reserved. # # This software is subject to the provisions of the Zope Public License, # Version 2.1 (ZPL). A copy of the ZPL should accompany this distribution. # THIS SOFTWARE IS PROVIDED "AS IS" AND ANY AND ALL EXPRESS OR IMPLIED # WARRANTIES ARE DISCLAIMED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED # WARRANTIES OF TITLE, MERCHANTABILITY, AGAINST INFRINGEMENT, AND FITNESS # FOR A PARTICULAR PURPOSE. # ############################################################################## """Restricted Python transformation examples This module contains pairs of functions. Each pair has a before and an after function. The after function shows the source code equivalent of the before function after it has been modified by the restricted compiler. These examples are actually used in the testRestrictions.py checkBeforeAndAfter() unit tests, which verifies that the restricted compiler actually produces the same output as would be output by the normal compiler for the after function. $Id: before_and_after.py 40218 2005-11-18 14:39:19Z andreasjung $ """ # getattr def simple_getattr_before(x): return x.y def simple_getattr_after(x): return _getattr_(x, 'y') # set attr def simple_setattr_before(): x.y = "bar" def simple_setattr_after(): _write_(x).y = "bar" # for loop and list comprehensions def simple_forloop_before(x): for x in [1, 2, 3]: pass def simple_forloop_after(x): for x in _getiter_([1, 2, 3]): pass def nested_forloop_before(x): for x in [1, 2, 3]: for y in "abc": pass def nested_forloop_after(x): for x in _getiter_([1, 2, 3]): for y in _getiter_("abc"): pass def simple_list_comprehension_before(): x = [y**2 for y in whatever if y > 3] def simple_list_comprehension_after(): x = [y**2 for y in _getiter_(whatever) if y > 3] def nested_list_comprehension_before(): x = [x**2 + y**2 for x in whatever1 if x >= 0 for y in whatever2 if y >= x] def nested_list_comprehension_after(): x = [x**2 + y**2 for x in _getiter_(whatever1) if x >= 0 for y in _getiter_(whatever2) if y >= x] # print def simple_print_before(): print "foo" def simple_print_after(): _print = _print_() print >> _print, "foo" # getitem def simple_getitem_before(): return x[0] def simple_getitem_after(): return _getitem_(x, 0) def simple_get_tuple_key_before(): x = y[1,2] def simple_get_tuple_key_after(): x = _getitem_(y, (1,2)) # set item def simple_setitem_before(): x[0] = "bar" def simple_setitem_after(): _write_(x)[0] = "bar" # delitem def simple_delitem_before(): del x[0] def simple_delitem_after(): del _write_(x)[0] # a collection of function parallels to many of the above def function_with_print_before(): def foo(): print "foo" return printed def function_with_print_after(): def foo(): _print = _print_() print >> _print, "foo" return _print() def function_with_getattr_before(): def foo(): return x.y def function_with_getattr_after(): def foo(): return _getattr_(x, 'y') def function_with_setattr_before(): def foo(x): x.y = "bar" def function_with_setattr_after(): def foo(x): _write_(x).y = "bar" def function_with_getitem_before(): def foo(x): return x[0] def function_with_getitem_after(): def foo(x): return _getitem_(x, 0) def function_with_forloop_before(): def foo(): for x in [1, 2, 3]: pass def function_with_forloop_after(): def foo(): for x in _getiter_([1, 2, 3]): pass # this, and all slices, won't work in these tests because the before code # parses the slice as a slice object, while the after code can't generate a # slice object in this way. The after code as written below # is parsed as a call to the 'slice' name, not as a slice object. # XXX solutions? #def simple_slice_before(): # x = y[:4] #def simple_slice_after(): # _getitem = _getitem_ # x = _getitem(y, slice(None, 4)) # Assignment stmts in Python can be very complicated. The "no_unpack" # test makes sure we're not doing unnecessary rewriting. def no_unpack_before(): x = y x = [y] x = y, x = (y, (y, y), [y, (y,)], x, (x, y)) x = y = z = (x, y, z) no_unpack_after = no_unpack_before # that is, should be untouched # apply() variations. Native apply() is unsafe because, e.g., # # def f(a, b, c): # whatever # # apply(f, two_element_sequence, dict_with_key_c) # # or (different spelling of the same thing) # # f(*two_element_sequence, **dict_with_key_c) # # makes the elements of two_element_sequence visible to f via its 'a' and # 'b' arguments, and the dict_with_key_c['c'] value visible via its 'c' # argument. That is, it's a devious way to extract values without going # thru security checks. def star_call_before(): foo(*a) def star_call_after(): _apply_(foo, *a) def star_call_2_before(): foo(0, *a) def star_call_2_after(): _apply_(foo, 0, *a) def starstar_call_before(): foo(**d) def starstar_call_after(): _apply_(foo, **d) def star_and_starstar_call_before(): foo(*a, **d) def star_and_starstar_call_after(): _apply_(foo, *a, **d) def positional_and_star_and_starstar_call_before(): foo(b, *a, **d) def positional_and_star_and_starstar_call_after(): _apply_(foo, b, *a, **d) def positional_and_defaults_and_star_and_starstar_call_before(): foo(b, x=y, w=z, *a, **d) def positional_and_defaults_and_star_and_starstar_call_after(): _apply_(foo, b, x=y, w=z, *a, **d) def lambda_with_getattr_in_defaults_before(): f = lambda x=y.z: x def lambda_with_getattr_in_defaults_after(): f = lambda x=_getattr_(y, "z"): x # augmented operators # Note that we don't have to worry about item, attr, or slice assignment, # as they are disallowed. Yay! ## def inplace_id_add_before(): ## x += y+z ## def inplace_id_add_after(): ## x = _inplacevar_('+=', x, y+z)