Welcome, guest | Sign In | My Account | Store | Cart
class Proxy(object):
    __slots__
= ["_obj", "__weakref__"]
   
def __init__(self, obj):
       
object.__setattr__(self, "_obj", obj)
   
   
#
   
# proxying (special cases)
   
#
   
def __getattribute__(self, name):
       
return getattr(object.__getattribute__(self, "_obj"), name)
   
def __delattr__(self, name):
        delattr
(object.__getattribute__(self, "_obj"), name)
   
def __setattr__(self, name, value):
        setattr
(object.__getattribute__(self, "_obj"), name, value)
   
   
def __nonzero__(self):
       
return bool(object.__getattribute__(self, "_obj"))
   
def __str__(self):
       
return str(object.__getattribute__(self, "_obj"))
   
def __repr__(self):
       
return repr(object.__getattribute__(self, "_obj"))
   
   
#
   
# factories
   
#
    _special_names
= [
       
'__abs__', '__add__', '__and__', '__call__', '__cmp__', '__coerce__',
       
'__contains__', '__delitem__', '__delslice__', '__div__', '__divmod__',
       
'__eq__', '__float__', '__floordiv__', '__ge__', '__getitem__',
       
'__getslice__', '__gt__', '__hash__', '__hex__', '__iadd__', '__iand__',
       
'__idiv__', '__idivmod__', '__ifloordiv__', '__ilshift__', '__imod__',
       
'__imul__', '__int__', '__invert__', '__ior__', '__ipow__', '__irshift__',
       
'__isub__', '__iter__', '__itruediv__', '__ixor__', '__le__', '__len__',
       
'__long__', '__lshift__', '__lt__', '__mod__', '__mul__', '__ne__',
       
'__neg__', '__oct__', '__or__', '__pos__', '__pow__', '__radd__',
       
'__rand__', '__rdiv__', '__rdivmod__', '__reduce__', '__reduce_ex__',
       
'__repr__', '__reversed__', '__rfloorfiv__', '__rlshift__', '__rmod__',
       
'__rmul__', '__ror__', '__rpow__', '__rrshift__', '__rshift__', '__rsub__',
       
'__rtruediv__', '__rxor__', '__setitem__', '__setslice__', '__sub__',
       
'__truediv__', '__xor__', 'next',
   
]
   
   
@classmethod
   
def _create_class_proxy(cls, theclass):
       
"""creates a proxy for the given class"""
       
       
def make_method(name):
           
def method(self, *args, **kw):
               
return getattr(object.__getattribute__(self, "_obj"), name)(*args, **kw)
           
return method
       
       
namespace = {}
       
for name in cls._special_names:
           
if hasattr(theclass, name):
               
namespace[name] = make_method(name)
       
return type("%s(%s)" % (cls.__name__, theclass.__name__), (cls,), namespace)
   
   
def __new__(cls, obj, *args, **kwargs):
       
"""
        creates an proxy instance referencing `obj`. (obj, *args, **kwargs) are
        passed to this class' __init__, so deriving classes can define an
        __init__ method of their own.
        note: _class_proxy_cache is unique per deriving class (each deriving
        class must hold its own cache)
        """

       
try:
            cache
= cls.__dict__["_class_proxy_cache"]
       
except KeyError:
            cls
._class_proxy_cache = cache = {}
       
try:
            theclass
= cache[obj.__class__]
       
except KeyError:
            cache
[obj.__class__] = theclass = cls._create_class_proxy(obj.__class__)
        ins
= object.__new__(theclass)
        theclass
.__init__(ins, obj, *args, **kwargs)
       
return ins


------ example ------
>>> p = Proxy(6)
>>> p
6
>>> type(p)
<class '__main__.Proxy(int)'>
>>> p + 2
8
>>> p2 = Proxy([1,2,3])
>>> p2
[1, 2, 3]
>>> dir(p2)
['__add__', '__class__', '__contains__', '__delattr__', '__delitem__ #...
'
__getslice__', '__gt__', '__hash__', '__iadd__', '__imul__', '__ini #...
educe__
', '__reduce_ex__', '__repr__', '__reversed__', '__rmul__', ' #...
'index', 'insert', 'pop', 'remove', 'reverse', 'sort']
>>> isinstance(p2, list)
True
>>> p2.append(8)
>>> p2.append(2)
>>> p2.append(5)
>>> p2
[1, 2, 3, 8, 2, 5]
>>> p2.sort()
>>> p2
[1, 2, 2, 3, 5, 8]
>>> p2[2]
2
>>> p2[-1]
8
>>> type(p2)
<class '__main__.Proxy(list)'>
>>> p2.__class__
<type 'list'>

----- exceptions -----
Proxying user-types should work perfectly well. But proxying builtin objects,
like ints
, floats, lists, etc., has some limitation and inconsistencies,
imposed
by the interpreter:

>>> p = Proxy(6)
>>> p + p
Traceback (most recent call last):
 
File "<stdin>", line 1, in ?
TypeError: unsupported operand type(s) for +: 'Proxy(int)' and 'Proxy(int)'


>>> Proxy([1,2,3]) + [4,5]
[1, 2, 3, 4, 5]
>>> Proxy([1,2,3]) + Proxy([4,5])
>>> p = Proxy([1,2,3])
>>> p.extend(Proxy([4,5]))
>>> p
[1, 2, 3, 4, 5]
>>> p + Proxy([6, 7])
Traceback (most recent call last):
 
File "<stdin>", line 1, in ?
 
File "Proxy.py", line 49, in method
   
return getattr(object.__getattribute__(self, "_obj"), name)(*args, **kw)
TypeError: can only concatenate list (not "Proxy(list)") to list


Also note that the methods of a proxied type return "real objects", not
proxies
. So,

>>> p = Proxy(3)
>>> type(p)
<class '__main__.Proxy(int)'>
>>> p + 1
4
>>> type(_)
<type 'int'>
>>> p += 1
>>> p
4
>>> type(p)
<type 'int'>

In this case, 'p' was reassigned a real integer, and the proxy was
garbage
-collected. What you might want to do is

>>> p = Proxy(3)
>>> p = Proxy(p + 1)
>>> p
4
>>> type(p)
<class '__main__.Proxy(int)'>

History