Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Proxy pattern in Python

I have a lots of class implemented in my code. Now I realise that for each method invoked for all these classes I need to add a line:

with service as object:

So I'm trying to use the Proxy pattern to automatically do the job, this is my example code

class A(object):
    def __init__(self, name):
        self.name = name
    def hello(self):
        print 'hello %s!' % (self.name)
    def __enter__(self):
        print 'Enter the function'
        return self
    def __exit__(self, exc_type, exc_value, traceback):
        print 'Exit the function'
class Proxy(object):
    def __init__(self, object_a):
#        object.__setattr__(self, '_object_a', object_a)
        self._object_a = object_a

    def __getattribute__(self, name):
        service = object.__getattribute__(self, '_object_a')
        with service as service:
            result = getattr(service, name)
        return result    

if __name__=='__main__':
    a1 = A('A1')
    b = Proxy(a1)
    b.hello()
    a2 = A('A2')
    b = Proxy(a2)
    b.hello()

Everything works find, I have the output:

Enter the function A1
Exit the function A1
hello A1!
Enter the function A2
Exit the function A2
hello A2!

But this is not exactly I need, because what I need is the equivalent of:

with a1 as a1:
    a1.hello()

And I must have the output:

Enter the function A1
hello A1!
Exit the function A1
Enter the function A2
hello A2!
Exit the function A2

What i need to have that result? Thanks

like image 294
nam Avatar asked Jun 04 '26 11:06

nam


1 Answers

I would use a decorator:

class Proxy(object):
    def __init__(self, object_a):
        self._object_a = object_a

    def decorateEnterExit(self, obj, f):
        def inner(*args, **kwargs):
            with obj as _:
                return f(*args, **kwargs)
        return inner

    def __getattribute__(self, name):
        obj = object.__getattribute__(self, '_object_a')
        dee = object.__getattribute__(self, 'decorateEnterExit')
        return dee(obj, getattr(obj, name))

That way the with will be only evaluated when the function is executed. The way you did it it would first evaluate the with (including exiting it) and then the function would be returned to be called. Now we return a function which will itself enter the with and call the function inside.

>>> Proxy(A('Ax')).hello()
Enter the function
hello Ax!
Exit the function

Note that you need to return self in your __enter__ method of A:

def __enter__(self):
    print 'Enter the function'
    return self
like image 164
poke Avatar answered Jun 07 '26 00:06

poke



Donate For Us

If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!