301 lines
7.0 KiB
Python
301 lines
7.0 KiB
Python
"""
|
|
Unit tests for slot-related functionality.
|
|
"""
|
|
|
|
import pytest
|
|
|
|
# Pympler doesn't work on PyPy.
|
|
try:
|
|
from pympler.asizeof import asizeof
|
|
has_pympler = True
|
|
except: # Won't be an import error.
|
|
has_pympler = False
|
|
|
|
import attr
|
|
|
|
|
|
@attr.s
|
|
class C1(object):
|
|
x = attr.ib(validator=attr.validators.instance_of(int))
|
|
y = attr.ib()
|
|
|
|
def method(self):
|
|
return self.x
|
|
|
|
@classmethod
|
|
def classmethod(cls):
|
|
return "clsmethod"
|
|
|
|
@staticmethod
|
|
def staticmethod():
|
|
return "staticmethod"
|
|
|
|
|
|
@attr.s(slots=True)
|
|
class C1Slots(object):
|
|
x = attr.ib(validator=attr.validators.instance_of(int))
|
|
y = attr.ib()
|
|
|
|
def method(self):
|
|
return self.x
|
|
|
|
@classmethod
|
|
def classmethod(cls):
|
|
return "clsmethod"
|
|
|
|
@staticmethod
|
|
def staticmethod():
|
|
return "staticmethod"
|
|
|
|
|
|
def test_slots_being_used():
|
|
"""
|
|
The class is really using __slots__.
|
|
"""
|
|
non_slot_instance = C1(x=1, y="test")
|
|
slot_instance = C1Slots(x=1, y="test")
|
|
|
|
assert "__dict__" not in dir(slot_instance)
|
|
assert "__slots__" in dir(slot_instance)
|
|
|
|
assert "__dict__" in dir(non_slot_instance)
|
|
assert "__slots__" not in dir(non_slot_instance)
|
|
|
|
assert set(["x", "y"]) == set(slot_instance.__slots__)
|
|
|
|
if has_pympler:
|
|
assert asizeof(slot_instance) < asizeof(non_slot_instance)
|
|
|
|
non_slot_instance.t = "test"
|
|
with pytest.raises(AttributeError):
|
|
slot_instance.t = "test"
|
|
|
|
assert 1 == non_slot_instance.method()
|
|
assert 1 == slot_instance.method()
|
|
|
|
assert attr.fields(C1Slots) == attr.fields(C1)
|
|
assert attr.asdict(slot_instance) == attr.asdict(non_slot_instance)
|
|
|
|
|
|
def test_basic_attr_funcs():
|
|
"""
|
|
Comparison, `__eq__`, `__hash__`, `__repr__`, `attrs.asdict` work.
|
|
"""
|
|
a = C1Slots(x=1, y=2)
|
|
b = C1Slots(x=1, y=3)
|
|
a_ = C1Slots(x=1, y=2)
|
|
|
|
# Comparison.
|
|
assert b > a
|
|
|
|
assert a_ == a
|
|
|
|
# Hashing.
|
|
hash(b) # Just to assert it doesn't raise.
|
|
|
|
# Repr.
|
|
assert "C1Slots(x=1, y=2)" == repr(a)
|
|
|
|
assert {"x": 1, "y": 2} == attr.asdict(a)
|
|
|
|
|
|
def test_inheritance_from_nonslots():
|
|
"""
|
|
Inheritance from a non-slot class works.
|
|
|
|
Note that a slots class inheriting from an ordinary class loses most of the
|
|
benefits of slots classes, but it should still work.
|
|
"""
|
|
@attr.s(slots=True)
|
|
class C2Slots(C1):
|
|
z = attr.ib()
|
|
|
|
c2 = C2Slots(x=1, y=2, z="test")
|
|
assert 1 == c2.x
|
|
assert 2 == c2.y
|
|
assert "test" == c2.z
|
|
c2.t = "test" # This will work, using the base class.
|
|
assert "test" == c2.t
|
|
|
|
assert 1 == c2.method()
|
|
assert "clsmethod" == c2.classmethod()
|
|
assert "staticmethod" == c2.staticmethod()
|
|
|
|
assert set(["z"]) == set(C2Slots.__slots__)
|
|
|
|
c3 = C2Slots(x=1, y=3, z="test")
|
|
assert c3 > c2
|
|
c2_ = C2Slots(x=1, y=2, z="test")
|
|
assert c2 == c2_
|
|
|
|
assert "C2Slots(x=1, y=2, z='test')" == repr(c2)
|
|
|
|
hash(c2) # Just to assert it doesn't raise.
|
|
|
|
assert {"x": 1, "y": 2, "z": "test"} == attr.asdict(c2)
|
|
|
|
|
|
def test_nonslots_these():
|
|
"""
|
|
Enhancing a non-slots class using 'these' works.
|
|
|
|
This will actually *replace* the class with another one, using slots.
|
|
"""
|
|
class SimpleOrdinaryClass(object):
|
|
def __init__(self, x, y, z):
|
|
self.x = x
|
|
self.y = y
|
|
self.z = z
|
|
|
|
def method(self):
|
|
return self.x
|
|
|
|
@classmethod
|
|
def classmethod(cls):
|
|
return "clsmethod"
|
|
|
|
@staticmethod
|
|
def staticmethod():
|
|
return "staticmethod"
|
|
|
|
C2Slots = attr.s(these={"x": attr.ib(), "y": attr.ib(), "z": attr.ib()},
|
|
init=False, slots=True)(SimpleOrdinaryClass)
|
|
|
|
c2 = C2Slots(x=1, y=2, z="test")
|
|
assert 1 == c2.x
|
|
assert 2 == c2.y
|
|
assert "test" == c2.z
|
|
with pytest.raises(AttributeError):
|
|
c2.t = "test" # We have slots now.
|
|
|
|
assert 1 == c2.method()
|
|
assert "clsmethod" == c2.classmethod()
|
|
assert "staticmethod" == c2.staticmethod()
|
|
|
|
assert set(["x", "y", "z"]) == set(C2Slots.__slots__)
|
|
|
|
c3 = C2Slots(x=1, y=3, z="test")
|
|
assert c3 > c2
|
|
c2_ = C2Slots(x=1, y=2, z="test")
|
|
assert c2 == c2_
|
|
|
|
assert "SimpleOrdinaryClass(x=1, y=2, z='test')" == repr(c2)
|
|
|
|
hash(c2) # Just to assert it doesn't raise.
|
|
|
|
assert {"x": 1, "y": 2, "z": "test"} == attr.asdict(c2)
|
|
|
|
|
|
def test_inheritance_from_slots():
|
|
"""
|
|
Inheriting from an attr slot class works.
|
|
"""
|
|
@attr.s(slots=True)
|
|
class C2Slots(C1Slots):
|
|
z = attr.ib()
|
|
|
|
@attr.s(slots=True)
|
|
class C2(C1):
|
|
z = attr.ib()
|
|
|
|
c2 = C2Slots(x=1, y=2, z="test")
|
|
assert 1 == c2.x
|
|
assert 2 == c2.y
|
|
assert "test" == c2.z
|
|
|
|
assert set(["z"]) == set(C2Slots.__slots__)
|
|
|
|
assert 1 == c2.method()
|
|
assert "clsmethod" == c2.classmethod()
|
|
assert "staticmethod" == c2.staticmethod()
|
|
|
|
with pytest.raises(AttributeError):
|
|
c2.t = "test"
|
|
|
|
non_slot_instance = C2(x=1, y=2, z="test")
|
|
if has_pympler:
|
|
assert asizeof(c2) < asizeof(non_slot_instance)
|
|
|
|
c3 = C2Slots(x=1, y=3, z="test")
|
|
assert c3 > c2
|
|
c2_ = C2Slots(x=1, y=2, z="test")
|
|
assert c2 == c2_
|
|
|
|
assert "C2Slots(x=1, y=2, z='test')" == repr(c2)
|
|
|
|
hash(c2) # Just to assert it doesn't raise.
|
|
|
|
assert {"x": 1, "y": 2, "z": "test"} == attr.asdict(c2)
|
|
|
|
|
|
def test_bare_inheritance_from_slots():
|
|
"""
|
|
Inheriting from a bare attr slot class works.
|
|
"""
|
|
@attr.s(init=False, cmp=False, hash=False, repr=False, slots=True)
|
|
class C1BareSlots(object):
|
|
x = attr.ib(validator=attr.validators.instance_of(int))
|
|
y = attr.ib()
|
|
|
|
def method(self):
|
|
return self.x
|
|
|
|
@classmethod
|
|
def classmethod(cls):
|
|
return "clsmethod"
|
|
|
|
@staticmethod
|
|
def staticmethod():
|
|
return "staticmethod"
|
|
|
|
@attr.s(init=False, cmp=False, hash=False, repr=False)
|
|
class C1Bare(object):
|
|
x = attr.ib(validator=attr.validators.instance_of(int))
|
|
y = attr.ib()
|
|
|
|
def method(self):
|
|
return self.x
|
|
|
|
@classmethod
|
|
def classmethod(cls):
|
|
return "clsmethod"
|
|
|
|
@staticmethod
|
|
def staticmethod():
|
|
return "staticmethod"
|
|
|
|
@attr.s(slots=True)
|
|
class C2Slots(C1BareSlots):
|
|
z = attr.ib()
|
|
|
|
@attr.s(slots=True)
|
|
class C2(C1Bare):
|
|
z = attr.ib()
|
|
|
|
c2 = C2Slots(x=1, y=2, z="test")
|
|
assert 1 == c2.x
|
|
assert 2 == c2.y
|
|
assert "test" == c2.z
|
|
|
|
assert 1 == c2.method()
|
|
assert "clsmethod" == c2.classmethod()
|
|
assert "staticmethod" == c2.staticmethod()
|
|
|
|
with pytest.raises(AttributeError):
|
|
c2.t = "test"
|
|
|
|
non_slot_instance = C2(x=1, y=2, z="test")
|
|
if has_pympler:
|
|
assert asizeof(c2) < asizeof(non_slot_instance)
|
|
|
|
c3 = C2Slots(x=1, y=3, z="test")
|
|
assert c3 > c2
|
|
c2_ = C2Slots(x=1, y=2, z="test")
|
|
assert c2 == c2_
|
|
|
|
assert "C2Slots(x=1, y=2, z='test')" == repr(c2)
|
|
|
|
hash(c2) # Just to assert it doesn't raise.
|
|
|
|
assert {"x": 1, "y": 2, "z": "test"} == attr.asdict(c2)
|