from pypy.conftest import gettestobjspace

class AppTest_Descriptor:

    OPTIONS = {}

    def setup_class(cls):
        cls.space = gettestobjspace(**cls.OPTIONS)

    def test_non_data_descr(self):
        class X(object):
            def f(self):
                return 42
        x = X()
        assert x.f() == 42
        x.f = 43
        assert x.f == 43
        del x.f
        assert x.f() == 42

    def test_set_without_get(self):
        class Descr(object):

            def __init__(self, name):
                self.name = name

            def __set__(self, obj, value):
                obj.__dict__[self.name] = value
        descr = Descr("a")

        class X(object):
            a = descr

        x = X()
        assert x.a is descr
        x.a = 42
        assert x.a == 42

    def test_failing_get(self):
        # when __get__() raises AttributeError,
        # __getattr__ is called...
        class X(object):
            def get_v(self):
                raise AttributeError
            v = property(get_v)

            def __getattr__(self, name):
                if name == 'v':
                    return 42
        x = X()
        assert x.v == 42

        # ... but the __dict__ is not searched
        class Y(object):
            def get_w(self):
                raise AttributeError
            def set_w(self, value):
                raise AttributeError
            w = property(get_w, set_w)
        y = Y()
        y.__dict__['w'] = 42
        raises(AttributeError, getattr, y, 'w')

    def test_member(self):
        class X(object):
            def __init__(self):
                self._v = 0
            def get_v(self):
                return self._v
            def set_v(self, v):
                self._v = v
            v = property(get_v, set_v)
        x = X()
        assert x.v  == 0
        assert X.v.__get__(x) == 0
        x.v = 1
        assert x.v == 1
        X.v.__set__(x, 0)
        assert x.v == 0
        raises(AttributeError, delattr, x, 'v')
        raises(AttributeError, X.v.__delete__, x)

    def test_special_methods_returning_strings(self): 
        class A(object): 
            seen = []
            def __str__(self): 
                self.seen.append(1) 
            def __repr__(self): 
                self.seen.append(2) 
            def __oct__(self): 
                self.seen.append(3) 
            def __hex__(self): 
                self.seen.append(4) 

        inst = A()
        raises(TypeError, str, inst) 
        raises(TypeError, repr, inst) 
        raises(TypeError, oct, inst) 
        raises(TypeError, hex, inst) 
        assert A.seen == [1,2,3,4]

    def test_hash(self): 
        class A(object):
            pass 
        hash(A()) 

        # as in CPython, for new-style classes we don't check if
        # __eq__ is overridden without __hash__ being overridden,
        # and so hash(B()) always just works (but gives a slightly
        # useless result).
        class B(object):
            def __eq__(self, other): pass 
        hash(B())

        # same as above for __cmp__
        class C(object):
            def __cmp__(self, other): pass 
        hash(C())

        class E(object):
            def __hash__(self): 
                return "something"
        raises(TypeError, hash, E())
        class F: # can return long
            def __hash__(self):
                return long(2**33)
        assert hash(F()) == hash(2**33) # 2.5 behavior

        class G:
            def __hash__(self):
                return 1
        assert isinstance(hash(G()), int)

        # __hash__ can return a subclass of long, but the fact that it's
        # a subclass is ignored
        class mylong(long):
            def __hash__(self):
                return 0
        class H(object):
            def __hash__(self):
                return mylong(42)
        assert hash(H()) == hash(42L)

        # don't return a subclass of int, either
        class myint(int):
            pass
        class I(object):
            def __hash__(self):
                return myint(15)
        assert hash(I()) == 15
        assert type(hash(I())) is int

