2015-01-29 19:50:42 +00:00
|
|
|
"""
|
|
|
|
Tests for `attr._make`.
|
|
|
|
"""
|
|
|
|
|
2015-01-27 16:53:17 +00:00
|
|
|
from __future__ import absolute_import, division, print_function
|
2017-02-04 13:37:49 +00:00
|
|
|
|
2018-01-17 12:16:39 +00:00
|
|
|
import copy
|
2017-02-04 13:37:49 +00:00
|
|
|
import inspect
|
2017-12-19 06:51:24 +00:00
|
|
|
import itertools
|
2017-11-04 13:38:03 +00:00
|
|
|
import sys
|
2017-02-04 13:37:49 +00:00
|
|
|
|
2016-11-19 08:47:03 +00:00
|
|
|
from operator import attrgetter
|
2015-01-27 16:53:17 +00:00
|
|
|
|
|
|
|
import pytest
|
2016-08-15 13:27:13 +00:00
|
|
|
|
2016-03-22 23:15:14 +00:00
|
|
|
from hypothesis import given
|
2016-11-19 08:47:03 +00:00
|
|
|
from hypothesis.strategies import booleans, integers, lists, sampled_from, text
|
2015-01-27 16:53:17 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
import attr
|
|
|
|
|
2015-02-20 10:30:46 +00:00
|
|
|
from attr import _config
|
2018-02-17 11:24:23 +00:00
|
|
|
from attr._compat import PY2, ordered_dict
|
2015-01-27 16:53:17 +00:00
|
|
|
from attr._make import (
|
2018-06-10 17:40:07 +00:00
|
|
|
Attribute,
|
|
|
|
Factory,
|
|
|
|
_AndValidator,
|
|
|
|
_Attributes,
|
|
|
|
_ClassBuilder,
|
|
|
|
_CountingAttr,
|
|
|
|
_transform_attrs,
|
|
|
|
and_,
|
|
|
|
fields,
|
|
|
|
fields_dict,
|
|
|
|
make_class,
|
|
|
|
validate,
|
2017-11-26 21:18:07 +00:00
|
|
|
)
|
|
|
|
from attr.exceptions import DefaultAlreadySetError, NotAnAttrsClassError
|
|
|
|
|
2018-04-10 22:00:57 +00:00
|
|
|
from .strategies import (
|
2018-06-10 17:40:07 +00:00
|
|
|
gen_attr_names,
|
|
|
|
list_of_attrs,
|
|
|
|
simple_attrs,
|
|
|
|
simple_attrs_with_metadata,
|
|
|
|
simple_attrs_without_metadata,
|
|
|
|
simple_classes,
|
2015-01-27 16:53:17 +00:00
|
|
|
)
|
2018-04-10 22:00:57 +00:00
|
|
|
from .utils import simple_attr
|
2015-01-27 16:53:17 +00:00
|
|
|
|
2016-08-15 13:27:13 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
attrs_st = simple_attrs.map(lambda c: Attribute.from_counting_attr("name", c))
|
2016-08-13 14:16:37 +00:00
|
|
|
|
2015-01-27 16:53:17 +00:00
|
|
|
|
2015-01-30 12:25:59 +00:00
|
|
|
class TestCountingAttr(object):
|
2015-01-27 16:53:17 +00:00
|
|
|
"""
|
2015-01-29 18:39:49 +00:00
|
|
|
Tests for `attr`.
|
2015-01-27 16:53:17 +00:00
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2015-01-27 16:53:17 +00:00
|
|
|
def test_returns_Attr(self):
|
|
|
|
"""
|
2015-01-29 19:50:42 +00:00
|
|
|
Returns an instance of _CountingAttr.
|
2015-01-27 16:53:17 +00:00
|
|
|
"""
|
2017-10-02 17:32:10 +00:00
|
|
|
a = attr.ib()
|
2017-02-11 15:55:39 +00:00
|
|
|
|
2015-01-27 16:53:17 +00:00
|
|
|
assert isinstance(a, _CountingAttr)
|
|
|
|
|
2017-02-11 15:55:39 +00:00
|
|
|
def test_validators_lists_to_wrapped_tuples(self):
|
|
|
|
"""
|
|
|
|
If a list is passed as validator, it's just converted to a tuple.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-02-11 15:55:39 +00:00
|
|
|
def v1(_, __):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def v2(_, __):
|
|
|
|
pass
|
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
a = attr.ib(validator=[v1, v2])
|
2017-02-11 15:55:39 +00:00
|
|
|
|
2018-06-10 17:40:07 +00:00
|
|
|
assert _AndValidator((v1, v2)) == a._validator
|
2017-02-11 15:55:39 +00:00
|
|
|
|
|
|
|
def test_validator_decorator_single(self):
|
|
|
|
"""
|
2017-05-12 21:02:07 +00:00
|
|
|
If _CountingAttr.validator is used as a decorator and there is no
|
|
|
|
decorator set, the decorated method is used as the validator.
|
2017-02-11 15:55:39 +00:00
|
|
|
"""
|
2017-10-02 17:32:10 +00:00
|
|
|
a = attr.ib()
|
2017-02-11 15:55:39 +00:00
|
|
|
|
|
|
|
@a.validator
|
|
|
|
def v():
|
|
|
|
pass
|
|
|
|
|
2017-05-12 21:02:07 +00:00
|
|
|
assert v == a._validator
|
2017-02-11 15:55:39 +00:00
|
|
|
|
2018-06-10 17:40:07 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"wrap", [lambda v: v, lambda v: [v], lambda v: and_(v)]
|
|
|
|
)
|
2017-05-12 21:02:07 +00:00
|
|
|
def test_validator_decorator(self, wrap):
|
2017-02-11 15:55:39 +00:00
|
|
|
"""
|
2017-05-12 21:02:07 +00:00
|
|
|
If _CountingAttr.validator is used as a decorator and there is already
|
|
|
|
a decorator set, the decorators are composed using `and_`.
|
2017-02-11 15:55:39 +00:00
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-02-11 15:55:39 +00:00
|
|
|
def v(_, __):
|
|
|
|
pass
|
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
a = attr.ib(validator=wrap(v))
|
2017-02-11 15:55:39 +00:00
|
|
|
|
|
|
|
@a.validator
|
|
|
|
def v2(self, _, __):
|
|
|
|
pass
|
|
|
|
|
2018-06-10 17:40:07 +00:00
|
|
|
assert _AndValidator((v, v2)) == a._validator
|
2017-02-11 15:55:39 +00:00
|
|
|
|
2017-05-16 07:36:39 +00:00
|
|
|
def test_default_decorator_already_set(self):
|
|
|
|
"""
|
|
|
|
Raise DefaultAlreadySetError if the decorator is used after a default
|
|
|
|
has been set.
|
|
|
|
"""
|
2017-10-02 17:32:10 +00:00
|
|
|
a = attr.ib(default=42)
|
2017-05-16 07:36:39 +00:00
|
|
|
|
|
|
|
with pytest.raises(DefaultAlreadySetError):
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-05-16 07:36:39 +00:00
|
|
|
@a.default
|
|
|
|
def f(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def test_default_decorator_sets(self):
|
|
|
|
"""
|
|
|
|
Decorator wraps the method in a Factory with pass_self=True and sets
|
|
|
|
the default.
|
|
|
|
"""
|
2017-10-02 17:32:10 +00:00
|
|
|
a = attr.ib()
|
2017-05-16 07:36:39 +00:00
|
|
|
|
|
|
|
@a.default
|
|
|
|
def f(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
assert Factory(f, True) == a._default
|
|
|
|
|
2015-01-27 16:53:17 +00:00
|
|
|
|
2017-12-23 07:46:10 +00:00
|
|
|
class TestAttribute(object):
|
|
|
|
"""
|
|
|
|
Tests for `attr.Attribute`.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-12-23 07:46:10 +00:00
|
|
|
def test_deprecated_convert_argument(self):
|
|
|
|
"""
|
|
|
|
Using *convert* raises a DeprecationWarning and sets the converter
|
|
|
|
field.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-12-23 07:46:10 +00:00
|
|
|
def conv(v):
|
|
|
|
return v
|
|
|
|
|
|
|
|
with pytest.warns(DeprecationWarning) as wi:
|
|
|
|
a = Attribute(
|
|
|
|
"a", True, True, True, True, True, True, convert=conv
|
|
|
|
)
|
|
|
|
w = wi.pop()
|
|
|
|
|
|
|
|
assert conv == a.converter
|
|
|
|
assert (
|
|
|
|
"The `convert` argument is deprecated in favor of `converter`. "
|
|
|
|
"It will be removed after 2019/01.",
|
|
|
|
) == w.message.args
|
|
|
|
assert __file__ == w.filename
|
|
|
|
|
|
|
|
def test_deprecated_convert_attribute(self):
|
|
|
|
"""
|
|
|
|
If Attribute.convert is accessed, a DeprecationWarning is raised.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-12-23 07:46:10 +00:00
|
|
|
def conv(v):
|
|
|
|
return v
|
|
|
|
|
|
|
|
a = simple_attr("a", converter=conv)
|
|
|
|
with pytest.warns(DeprecationWarning) as wi:
|
|
|
|
convert = a.convert
|
|
|
|
w = wi.pop()
|
|
|
|
|
|
|
|
assert conv is convert is a.converter
|
|
|
|
assert (
|
|
|
|
"The `convert` attribute is deprecated in favor of `converter`. "
|
|
|
|
"It will be removed after 2019/01.",
|
|
|
|
) == w.message.args
|
|
|
|
assert __file__ == w.filename
|
|
|
|
|
|
|
|
def test_convert_converter(self):
|
|
|
|
"""
|
|
|
|
A TypeError is raised if both *convert* and *converter* are passed.
|
|
|
|
"""
|
|
|
|
with pytest.raises(RuntimeError) as ei:
|
|
|
|
Attribute(
|
2018-06-10 17:40:07 +00:00
|
|
|
"a",
|
|
|
|
True,
|
|
|
|
True,
|
|
|
|
True,
|
|
|
|
True,
|
|
|
|
True,
|
|
|
|
True,
|
|
|
|
convert=lambda v: v,
|
|
|
|
converter=lambda v: v,
|
2017-12-23 07:46:10 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
assert (
|
|
|
|
"Can't pass both `convert` and `converter`. "
|
|
|
|
"Please use `converter` only.",
|
|
|
|
) == ei.value.args
|
|
|
|
|
|
|
|
|
2015-01-29 15:24:49 +00:00
|
|
|
def make_tc():
|
|
|
|
class TransformC(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
z = attr.ib()
|
|
|
|
y = attr.ib()
|
|
|
|
x = attr.ib()
|
2015-01-29 15:24:49 +00:00
|
|
|
a = 42
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2015-01-29 15:24:49 +00:00
|
|
|
return TransformC
|
|
|
|
|
|
|
|
|
|
|
|
class TestTransformAttrs(object):
|
2015-01-27 16:53:17 +00:00
|
|
|
"""
|
2015-01-29 15:24:49 +00:00
|
|
|
Tests for `_transform_attrs`.
|
2015-01-27 16:53:17 +00:00
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-26 10:55:34 +00:00
|
|
|
def test_no_modifications(self):
|
|
|
|
"""
|
|
|
|
Doesn't attach __attrs_attrs__ to the class anymore.
|
|
|
|
"""
|
|
|
|
C = make_tc()
|
2017-11-08 10:15:21 +00:00
|
|
|
_transform_attrs(C, None, False)
|
2017-10-26 10:55:34 +00:00
|
|
|
|
|
|
|
assert None is getattr(C, "__attrs_attrs__", None)
|
|
|
|
|
2015-01-27 16:53:17 +00:00
|
|
|
def test_normal(self):
|
|
|
|
"""
|
2015-01-29 15:24:49 +00:00
|
|
|
Transforms every `_CountingAttr` and leaves others (a) be.
|
2015-01-27 16:53:17 +00:00
|
|
|
"""
|
2015-01-29 15:24:49 +00:00
|
|
|
C = make_tc()
|
2018-01-29 06:35:01 +00:00
|
|
|
attrs, _, _ = _transform_attrs(C, None, False)
|
2017-10-26 10:55:34 +00:00
|
|
|
|
|
|
|
assert ["z", "y", "x"] == [a.name for a in attrs]
|
2015-01-27 16:53:17 +00:00
|
|
|
|
|
|
|
def test_empty(self):
|
|
|
|
"""
|
2015-01-29 15:24:49 +00:00
|
|
|
No attributes works as expected.
|
2015-01-27 16:53:17 +00:00
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s
|
2015-01-27 16:53:17 +00:00
|
|
|
class C(object):
|
|
|
|
pass
|
|
|
|
|
2018-01-29 06:35:01 +00:00
|
|
|
assert _Attributes(((), [], {})) == _transform_attrs(C, None, False)
|
2015-01-29 15:24:49 +00:00
|
|
|
|
2017-10-26 10:55:34 +00:00
|
|
|
def test_transforms_to_attribute(self):
|
2015-01-29 15:24:49 +00:00
|
|
|
"""
|
|
|
|
All `_CountingAttr`s are transformed into `Attribute`s.
|
|
|
|
"""
|
|
|
|
C = make_tc()
|
2018-01-29 06:35:01 +00:00
|
|
|
attrs, super_attrs, _ = _transform_attrs(C, None, False)
|
2015-01-29 15:24:49 +00:00
|
|
|
|
2017-10-26 10:55:34 +00:00
|
|
|
assert [] == super_attrs
|
|
|
|
assert 3 == len(attrs)
|
|
|
|
assert all(isinstance(a, Attribute) for a in attrs)
|
2015-01-27 16:53:17 +00:00
|
|
|
|
2015-01-29 19:50:42 +00:00
|
|
|
def test_conflicting_defaults(self):
|
|
|
|
"""
|
|
|
|
Raises `ValueError` if attributes with defaults are followed by
|
|
|
|
mandatory attributes.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2015-01-29 19:50:42 +00:00
|
|
|
class C(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
x = attr.ib(default=None)
|
|
|
|
y = attr.ib()
|
2015-01-29 19:50:42 +00:00
|
|
|
|
|
|
|
with pytest.raises(ValueError) as e:
|
2017-11-08 10:15:21 +00:00
|
|
|
_transform_attrs(C, None, False)
|
2015-01-29 19:50:42 +00:00
|
|
|
assert (
|
2015-02-07 20:03:17 +00:00
|
|
|
"No mandatory attributes allowed after an attribute with a "
|
2015-01-29 19:50:42 +00:00
|
|
|
"default value or factory. Attribute in question: Attribute"
|
2015-02-20 12:29:47 +00:00
|
|
|
"(name='y', default=NOTHING, validator=None, repr=True, "
|
2017-12-23 07:46:10 +00:00
|
|
|
"cmp=True, hash=None, init=True, metadata=mappingproxy({}), "
|
|
|
|
"type=None, converter=None)",
|
2015-01-29 19:50:42 +00:00
|
|
|
) == e.value.args
|
|
|
|
|
2015-02-08 11:32:32 +00:00
|
|
|
def test_these(self):
|
|
|
|
"""
|
2017-10-26 10:55:34 +00:00
|
|
|
If these is passed, use it and ignore body and super classes.
|
2015-02-08 11:32:32 +00:00
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-26 10:55:34 +00:00
|
|
|
class Base(object):
|
|
|
|
z = attr.ib()
|
2015-02-18 20:31:32 +00:00
|
|
|
|
2017-10-26 10:55:34 +00:00
|
|
|
class C(Base):
|
|
|
|
y = attr.ib()
|
2015-08-20 11:25:32 +00:00
|
|
|
|
2018-01-29 06:35:01 +00:00
|
|
|
attrs, super_attrs, _ = _transform_attrs(C, {"x": attr.ib()}, False)
|
2015-08-20 11:25:32 +00:00
|
|
|
|
2017-10-26 10:55:34 +00:00
|
|
|
assert [] == super_attrs
|
2018-06-10 17:40:07 +00:00
|
|
|
assert (simple_attr("x"),) == attrs
|
2015-02-18 20:31:32 +00:00
|
|
|
|
2018-01-16 18:09:23 +00:00
|
|
|
def test_these_leave_body(self):
|
|
|
|
"""
|
|
|
|
If these is passed, no attributes are removed from the body.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2018-01-16 18:09:23 +00:00
|
|
|
@attr.s(init=False, these={"x": attr.ib()})
|
|
|
|
class C(object):
|
|
|
|
x = 5
|
|
|
|
|
|
|
|
assert 5 == C().x
|
|
|
|
assert "C(x=5)" == repr(C())
|
|
|
|
|
2018-02-17 11:24:23 +00:00
|
|
|
def test_these_ordered(self):
|
|
|
|
"""
|
|
|
|
If these is passed ordered attrs, their order respect instead of the
|
|
|
|
counter.
|
|
|
|
"""
|
|
|
|
b = attr.ib(default=2)
|
|
|
|
a = attr.ib(default=1)
|
|
|
|
|
|
|
|
@attr.s(these=ordered_dict([("a", a), ("b", b)]))
|
|
|
|
class C(object):
|
|
|
|
pass
|
|
|
|
|
|
|
|
assert "C(a=1, b=2)" == repr(C())
|
|
|
|
|
2017-11-11 06:09:17 +00:00
|
|
|
def test_multiple_inheritance(self):
|
|
|
|
"""
|
|
|
|
Order of attributes doesn't get mixed up by multiple inheritance.
|
|
|
|
|
|
|
|
See #285
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-11-11 06:09:17 +00:00
|
|
|
@attr.s
|
|
|
|
class A(object):
|
|
|
|
a1 = attr.ib(default="a1")
|
|
|
|
a2 = attr.ib(default="a2")
|
|
|
|
|
|
|
|
@attr.s
|
|
|
|
class B(A):
|
|
|
|
b1 = attr.ib(default="b1")
|
|
|
|
b2 = attr.ib(default="b2")
|
|
|
|
|
|
|
|
@attr.s
|
|
|
|
class C(B, A):
|
|
|
|
c1 = attr.ib(default="c1")
|
|
|
|
c2 = attr.ib(default="c2")
|
|
|
|
|
|
|
|
@attr.s
|
|
|
|
class D(A):
|
|
|
|
d1 = attr.ib(default="d1")
|
|
|
|
d2 = attr.ib(default="d2")
|
|
|
|
|
|
|
|
@attr.s
|
2017-12-13 14:33:44 +00:00
|
|
|
class E(C, D):
|
2017-11-11 06:09:17 +00:00
|
|
|
e1 = attr.ib(default="e1")
|
|
|
|
e2 = attr.ib(default="e2")
|
|
|
|
|
|
|
|
assert (
|
|
|
|
"E(a1='a1', a2='a2', b1='b1', b2='b2', c1='c1', c2='c2', d1='d1', "
|
|
|
|
"d2='d2', e1='e1', e2='e2')"
|
|
|
|
) == repr(E())
|
|
|
|
|
2015-01-27 16:53:17 +00:00
|
|
|
|
2015-01-29 19:50:42 +00:00
|
|
|
class TestAttributes(object):
|
2015-01-27 16:53:17 +00:00
|
|
|
"""
|
2017-02-19 08:51:43 +00:00
|
|
|
Tests for the `attrs`/`attr.s` class decorator.
|
2015-01-27 16:53:17 +00:00
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2016-08-16 10:35:27 +00:00
|
|
|
@pytest.mark.skipif(not PY2, reason="No old-style classes in Py3")
|
2015-02-18 19:38:29 +00:00
|
|
|
def test_catches_old_style(self):
|
|
|
|
"""
|
|
|
|
Raises TypeError on old-style classes.
|
|
|
|
"""
|
|
|
|
with pytest.raises(TypeError) as e:
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s
|
2015-02-18 19:38:29 +00:00
|
|
|
class C:
|
|
|
|
pass
|
2017-10-26 10:55:34 +00:00
|
|
|
|
2015-02-18 19:38:29 +00:00
|
|
|
assert ("attrs only works with new-style classes.",) == e.value.args
|
|
|
|
|
2015-01-27 16:53:17 +00:00
|
|
|
def test_sets_attrs(self):
|
|
|
|
"""
|
|
|
|
Sets the `__attrs_attrs__` class attribute with a list of `Attribute`s.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s
|
2015-01-27 16:53:17 +00:00
|
|
|
class C(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
x = attr.ib()
|
2017-10-26 10:55:34 +00:00
|
|
|
|
2015-01-27 16:53:17 +00:00
|
|
|
assert "x" == C.__attrs_attrs__[0].name
|
|
|
|
assert all(isinstance(a, Attribute) for a in C.__attrs_attrs__)
|
|
|
|
|
|
|
|
def test_empty(self):
|
|
|
|
"""
|
|
|
|
No attributes, no problems.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s
|
2015-01-27 16:53:17 +00:00
|
|
|
class C3(object):
|
|
|
|
pass
|
2017-10-26 10:55:34 +00:00
|
|
|
|
2015-01-27 22:08:55 +00:00
|
|
|
assert "C3()" == repr(C3())
|
2015-01-27 16:53:17 +00:00
|
|
|
assert C3() == C3()
|
2015-01-29 15:24:49 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@given(attr=attrs_st, attr_name=sampled_from(Attribute.__slots__))
|
2016-08-13 14:16:37 +00:00
|
|
|
def test_immutable(self, attr, attr_name):
|
|
|
|
"""
|
|
|
|
Attribute instances are immutable.
|
|
|
|
"""
|
|
|
|
with pytest.raises(AttributeError):
|
|
|
|
setattr(attr, attr_name, 1)
|
|
|
|
|
2018-06-10 17:40:07 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"method_name", ["__repr__", "__eq__", "__hash__", "__init__"]
|
|
|
|
)
|
2015-01-29 15:24:49 +00:00
|
|
|
def test_adds_all_by_default(self, method_name):
|
|
|
|
"""
|
2017-02-19 08:51:43 +00:00
|
|
|
If no further arguments are supplied, all add_XXX functions except
|
|
|
|
add_hash are applied. __hash__ is set to None.
|
2015-01-29 15:24:49 +00:00
|
|
|
"""
|
|
|
|
# Set the method name to a sentinel and check whether it has been
|
|
|
|
# overwritten afterwards.
|
|
|
|
sentinel = object()
|
|
|
|
|
2017-02-19 08:51:43 +00:00
|
|
|
class C(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
x = attr.ib()
|
2015-01-29 15:24:49 +00:00
|
|
|
|
2017-02-19 08:51:43 +00:00
|
|
|
setattr(C, method_name, sentinel)
|
2015-01-29 15:24:49 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
C = attr.s(C)
|
2017-02-19 08:51:43 +00:00
|
|
|
meth = getattr(C, method_name)
|
2015-01-29 15:24:49 +00:00
|
|
|
|
2017-02-19 08:51:43 +00:00
|
|
|
assert sentinel != meth
|
|
|
|
if method_name == "__hash__":
|
|
|
|
assert meth is None
|
2015-01-29 15:24:49 +00:00
|
|
|
|
2018-06-10 17:40:07 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"arg_name, method_name",
|
|
|
|
[
|
|
|
|
("repr", "__repr__"),
|
|
|
|
("cmp", "__eq__"),
|
|
|
|
("hash", "__hash__"),
|
|
|
|
("init", "__init__"),
|
|
|
|
],
|
|
|
|
)
|
2015-01-29 15:24:49 +00:00
|
|
|
def test_respects_add_arguments(self, arg_name, method_name):
|
|
|
|
"""
|
2017-02-19 08:51:43 +00:00
|
|
|
If a certain `add_XXX` is `False`, `__XXX__` is not added to the class.
|
2015-01-29 15:24:49 +00:00
|
|
|
"""
|
|
|
|
# Set the method name to a sentinel and check whether it has been
|
|
|
|
# overwritten afterwards.
|
|
|
|
sentinel = object()
|
|
|
|
|
2018-06-10 17:40:07 +00:00
|
|
|
am_args = {"repr": True, "cmp": True, "hash": True, "init": True}
|
2015-02-20 12:29:47 +00:00
|
|
|
am_args[arg_name] = False
|
2015-01-29 15:24:49 +00:00
|
|
|
|
|
|
|
class C(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
x = attr.ib()
|
2015-01-29 15:24:49 +00:00
|
|
|
|
|
|
|
setattr(C, method_name, sentinel)
|
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
C = attr.s(**am_args)(C)
|
2015-01-29 15:24:49 +00:00
|
|
|
|
|
|
|
assert sentinel == getattr(C, method_name)
|
|
|
|
|
2016-08-16 10:35:27 +00:00
|
|
|
@pytest.mark.skipif(PY2, reason="__qualname__ is PY3-only.")
|
2016-03-22 23:15:14 +00:00
|
|
|
@given(slots_outer=booleans(), slots_inner=booleans())
|
2016-03-16 21:28:43 +00:00
|
|
|
def test_repr_qualname(self, slots_outer, slots_inner):
|
2015-02-20 08:44:50 +00:00
|
|
|
"""
|
|
|
|
On Python 3, the name in repr is the __qualname__.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s(slots=slots_outer)
|
2015-02-20 08:44:50 +00:00
|
|
|
class C(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s(slots=slots_inner)
|
2015-02-20 08:44:50 +00:00
|
|
|
class D(object):
|
|
|
|
pass
|
|
|
|
|
|
|
|
assert "C.D()" == repr(C.D())
|
|
|
|
assert "GC.D()" == repr(GC.D())
|
|
|
|
|
2016-03-22 23:15:14 +00:00
|
|
|
@given(slots_outer=booleans(), slots_inner=booleans())
|
2016-03-16 21:28:43 +00:00
|
|
|
def test_repr_fake_qualname(self, slots_outer, slots_inner):
|
2015-02-20 08:44:50 +00:00
|
|
|
"""
|
|
|
|
Setting repr_ns overrides a potentially guessed namespace.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s(slots=slots_outer)
|
2015-02-20 08:44:50 +00:00
|
|
|
class C(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s(repr_ns="C", slots=slots_inner)
|
2015-02-20 08:44:50 +00:00
|
|
|
class D(object):
|
|
|
|
pass
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2015-02-20 08:44:50 +00:00
|
|
|
assert "C.D()" == repr(C.D())
|
|
|
|
|
2016-09-30 13:48:31 +00:00
|
|
|
@pytest.mark.skipif(PY2, reason="__qualname__ is PY3-only.")
|
|
|
|
@given(slots_outer=booleans(), slots_inner=booleans())
|
|
|
|
def test_name_not_overridden(self, slots_outer, slots_inner):
|
|
|
|
"""
|
|
|
|
On Python 3, __name__ is different from __qualname__.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s(slots=slots_outer)
|
2016-09-30 13:48:31 +00:00
|
|
|
class C(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s(slots=slots_inner)
|
2016-09-30 13:48:31 +00:00
|
|
|
class D(object):
|
|
|
|
pass
|
|
|
|
|
|
|
|
assert C.D.__name__ == "D"
|
|
|
|
assert C.D.__qualname__ == C.__qualname__ + ".D"
|
|
|
|
|
2017-01-06 14:08:17 +00:00
|
|
|
@given(with_validation=booleans())
|
|
|
|
def test_post_init(self, with_validation, monkeypatch):
|
2016-11-11 04:16:48 +00:00
|
|
|
"""
|
2016-11-20 13:39:34 +00:00
|
|
|
Verify that __attrs_post_init__ gets called if defined.
|
2016-11-11 04:16:48 +00:00
|
|
|
"""
|
2017-02-11 15:55:39 +00:00
|
|
|
monkeypatch.setattr(_config, "_run_validators", with_validation)
|
2017-01-06 14:08:17 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s
|
2016-11-11 04:16:48 +00:00
|
|
|
class C(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
x = attr.ib()
|
|
|
|
y = attr.ib()
|
2016-11-11 04:16:48 +00:00
|
|
|
|
2016-11-20 13:39:34 +00:00
|
|
|
def __attrs_post_init__(self2):
|
2016-11-11 04:16:48 +00:00
|
|
|
self2.z = self2.x + self2.y
|
|
|
|
|
|
|
|
c = C(x=10, y=20)
|
2017-10-02 10:14:37 +00:00
|
|
|
|
2018-06-10 17:40:07 +00:00
|
|
|
assert 30 == getattr(c, "z", None)
|
2016-11-11 04:16:48 +00:00
|
|
|
|
2017-09-17 14:22:49 +00:00
|
|
|
def test_types(self):
|
|
|
|
"""
|
|
|
|
Sets the `Attribute.type` attr from type argument.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s
|
2017-09-17 14:22:49 +00:00
|
|
|
class C(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
x = attr.ib(type=int)
|
|
|
|
y = attr.ib(type=str)
|
|
|
|
z = attr.ib()
|
2017-10-02 10:14:37 +00:00
|
|
|
|
2017-09-17 14:22:49 +00:00
|
|
|
assert int is fields(C).x.type
|
|
|
|
assert str is fields(C).y.type
|
|
|
|
assert None is fields(C).z.type
|
|
|
|
|
2017-10-02 10:14:37 +00:00
|
|
|
@pytest.mark.parametrize("slots", [True, False])
|
|
|
|
def test_clean_class(self, slots):
|
|
|
|
"""
|
|
|
|
Attribute definitions do not appear on the class body after @attr.s.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s(slots=slots)
|
2017-10-02 10:14:37 +00:00
|
|
|
class C(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
x = attr.ib()
|
2017-10-02 10:14:37 +00:00
|
|
|
|
|
|
|
x = getattr(C, "x", None)
|
|
|
|
|
|
|
|
assert not isinstance(x, _CountingAttr)
|
|
|
|
|
2018-03-14 18:05:48 +00:00
|
|
|
def test_factory_sugar(self):
|
|
|
|
"""
|
|
|
|
Passing factory=f is syntactic sugar for passing default=Factory(f).
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2018-03-14 18:05:48 +00:00
|
|
|
@attr.s
|
|
|
|
class C(object):
|
|
|
|
x = attr.ib(factory=list)
|
|
|
|
|
|
|
|
assert Factory(list) == attr.fields(C).x.default
|
|
|
|
|
|
|
|
def test_sugar_factory_mutex(self):
|
|
|
|
"""
|
|
|
|
Passing both default and factory raises ValueError.
|
|
|
|
"""
|
|
|
|
with pytest.raises(ValueError, match="mutually exclusive"):
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2018-03-14 18:05:48 +00:00
|
|
|
@attr.s
|
|
|
|
class C(object):
|
|
|
|
x = attr.ib(factory=list, default=Factory(list))
|
|
|
|
|
|
|
|
def test_sugar_callable(self):
|
|
|
|
"""
|
|
|
|
Factory has to be a callable to prevent people from passing Factory
|
|
|
|
into it.
|
|
|
|
"""
|
|
|
|
with pytest.raises(ValueError, match="must be a callable"):
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2018-03-14 18:05:48 +00:00
|
|
|
@attr.s
|
|
|
|
class C(object):
|
|
|
|
x = attr.ib(factory=Factory(list))
|
|
|
|
|
2015-02-20 08:44:50 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s
|
2015-02-20 08:44:50 +00:00
|
|
|
class GC(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s
|
2015-02-20 08:44:50 +00:00
|
|
|
class D(object):
|
|
|
|
pass
|
|
|
|
|
2015-01-29 15:24:49 +00:00
|
|
|
|
2015-01-30 07:57:33 +00:00
|
|
|
class TestMakeClass(object):
|
|
|
|
"""
|
|
|
|
Tests for `make_class`.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
|
|
|
@pytest.mark.parametrize("ls", [list, tuple])
|
2015-01-30 07:57:33 +00:00
|
|
|
def test_simple(self, ls):
|
|
|
|
"""
|
|
|
|
Passing a list of strings creates attributes with default args.
|
|
|
|
"""
|
|
|
|
C1 = make_class("C1", ls(["a", "b"]))
|
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s
|
2015-01-30 07:57:33 +00:00
|
|
|
class C2(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
a = attr.ib()
|
|
|
|
b = attr.ib()
|
2015-01-30 07:57:33 +00:00
|
|
|
|
|
|
|
assert C1.__attrs_attrs__ == C2.__attrs_attrs__
|
|
|
|
|
|
|
|
def test_dict(self):
|
|
|
|
"""
|
|
|
|
Passing a dict of name: _CountingAttr creates an equivalent class.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
C1 = make_class(
|
|
|
|
"C1", {"a": attr.ib(default=42), "b": attr.ib(default=None)}
|
|
|
|
)
|
2015-01-30 07:57:33 +00:00
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
@attr.s
|
2015-01-30 07:57:33 +00:00
|
|
|
class C2(object):
|
2017-10-02 17:32:10 +00:00
|
|
|
a = attr.ib(default=42)
|
|
|
|
b = attr.ib(default=None)
|
2015-01-30 07:57:33 +00:00
|
|
|
|
|
|
|
assert C1.__attrs_attrs__ == C2.__attrs_attrs__
|
|
|
|
|
|
|
|
def test_attr_args(self):
|
|
|
|
"""
|
|
|
|
attributes_arguments are passed to attributes
|
|
|
|
"""
|
2015-02-20 12:29:47 +00:00
|
|
|
C = make_class("C", ["x"], repr=False)
|
2017-11-04 13:43:29 +00:00
|
|
|
|
2017-11-04 13:38:03 +00:00
|
|
|
assert repr(C(1)).startswith("<tests.test_make.C object at 0x")
|
2015-01-30 07:57:33 +00:00
|
|
|
|
|
|
|
def test_catches_wrong_attrs_type(self):
|
|
|
|
"""
|
|
|
|
Raise `TypeError` if an invalid type for attrs is passed.
|
|
|
|
"""
|
|
|
|
with pytest.raises(TypeError) as e:
|
|
|
|
make_class("C", object())
|
|
|
|
|
2018-06-10 17:40:07 +00:00
|
|
|
assert ("attrs argument must be a dict or a list.",) == e.value.args
|
2015-02-09 12:16:56 +00:00
|
|
|
|
2017-02-21 06:44:59 +00:00
|
|
|
def test_bases(self):
|
|
|
|
"""
|
|
|
|
Parameter bases default to (object,) and subclasses correctly
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-02-21 06:44:59 +00:00
|
|
|
class D(object):
|
|
|
|
pass
|
|
|
|
|
|
|
|
cls = make_class("C", {})
|
2017-11-04 13:43:29 +00:00
|
|
|
|
2017-02-21 06:44:59 +00:00
|
|
|
assert cls.__mro__[-1] == object
|
|
|
|
|
|
|
|
cls = make_class("C", {}, bases=(D,))
|
2017-11-04 13:43:29 +00:00
|
|
|
|
2017-02-21 06:44:59 +00:00
|
|
|
assert D in cls.__mro__
|
|
|
|
assert isinstance(cls(), D)
|
|
|
|
|
2017-10-02 10:14:37 +00:00
|
|
|
@pytest.mark.parametrize("slots", [True, False])
|
|
|
|
def test_clean_class(self, slots):
|
|
|
|
"""
|
|
|
|
Attribute definitions do not appear on the class body.
|
|
|
|
"""
|
|
|
|
C = make_class("C", ["x"], slots=slots)
|
|
|
|
|
|
|
|
x = getattr(C, "x", None)
|
|
|
|
|
|
|
|
assert not isinstance(x, _CountingAttr)
|
|
|
|
|
2017-11-04 13:38:03 +00:00
|
|
|
def test_missing_sys_getframe(self, monkeypatch):
|
|
|
|
"""
|
|
|
|
`make_class()` does not fail when `sys._getframe()` is not available.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
monkeypatch.delattr(sys, "_getframe")
|
2017-11-04 13:38:03 +00:00
|
|
|
C = make_class("C", ["x"])
|
2017-11-04 13:43:29 +00:00
|
|
|
|
|
|
|
assert 1 == len(C.__attrs_attrs__)
|
2017-11-04 13:38:03 +00:00
|
|
|
|
2018-02-17 11:24:23 +00:00
|
|
|
def test_make_class_ordered(self):
|
|
|
|
"""
|
|
|
|
If `make_class()` is passed ordered attrs, their order is respected
|
|
|
|
instead of the counter.
|
|
|
|
"""
|
|
|
|
b = attr.ib(default=2)
|
|
|
|
a = attr.ib(default=1)
|
|
|
|
|
|
|
|
C = attr.make_class("C", ordered_dict([("a", a), ("b", b)]))
|
|
|
|
|
|
|
|
assert "C(a=1, b=2)" == repr(C())
|
|
|
|
|
2015-02-09 12:16:56 +00:00
|
|
|
|
|
|
|
class TestFields(object):
|
|
|
|
"""
|
|
|
|
Tests for `fields`.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2015-02-09 12:16:56 +00:00
|
|
|
def test_instance(self, C):
|
|
|
|
"""
|
|
|
|
Raises `TypeError` on non-classes.
|
|
|
|
"""
|
|
|
|
with pytest.raises(TypeError) as e:
|
|
|
|
fields(C(1, 2))
|
2017-11-04 13:43:29 +00:00
|
|
|
|
2015-02-09 12:16:56 +00:00
|
|
|
assert "Passed object must be a class." == e.value.args[0]
|
|
|
|
|
|
|
|
def test_handler_non_attrs_class(self, C):
|
|
|
|
"""
|
|
|
|
Raises `ValueError` if passed a non-``attrs`` instance.
|
|
|
|
"""
|
2016-09-10 17:14:34 +00:00
|
|
|
with pytest.raises(NotAnAttrsClassError) as e:
|
2015-02-09 12:16:56 +00:00
|
|
|
fields(object)
|
2018-03-03 14:09:05 +00:00
|
|
|
|
2015-02-09 12:16:56 +00:00
|
|
|
assert (
|
|
|
|
"{o!r} is not an attrs-decorated class.".format(o=object)
|
|
|
|
) == e.value.args[0]
|
|
|
|
|
2016-09-10 15:50:17 +00:00
|
|
|
@given(simple_classes())
|
2015-02-09 12:16:56 +00:00
|
|
|
def test_fields(self, C):
|
|
|
|
"""
|
|
|
|
Returns a list of `Attribute`a.
|
|
|
|
"""
|
|
|
|
assert all(isinstance(a, Attribute) for a in fields(C))
|
|
|
|
|
2016-09-10 15:50:17 +00:00
|
|
|
@given(simple_classes())
|
|
|
|
def test_fields_properties(self, C):
|
|
|
|
"""
|
|
|
|
Fields returns a tuple with properties.
|
|
|
|
"""
|
|
|
|
for attribute in fields(C):
|
|
|
|
assert getattr(fields(C), attribute.name) is attribute
|
|
|
|
|
2015-02-09 12:16:56 +00:00
|
|
|
|
2018-03-03 14:09:05 +00:00
|
|
|
class TestFieldsDict(object):
|
|
|
|
"""
|
|
|
|
Tests for `fields_dict`.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2018-03-03 14:09:05 +00:00
|
|
|
def test_instance(self, C):
|
|
|
|
"""
|
|
|
|
Raises `TypeError` on non-classes.
|
|
|
|
"""
|
|
|
|
with pytest.raises(TypeError) as e:
|
|
|
|
fields_dict(C(1, 2))
|
|
|
|
|
|
|
|
assert "Passed object must be a class." == e.value.args[0]
|
|
|
|
|
|
|
|
def test_handler_non_attrs_class(self, C):
|
|
|
|
"""
|
|
|
|
Raises `ValueError` if passed a non-``attrs`` instance.
|
|
|
|
"""
|
|
|
|
with pytest.raises(NotAnAttrsClassError) as e:
|
|
|
|
fields_dict(object)
|
|
|
|
|
|
|
|
assert (
|
|
|
|
"{o!r} is not an attrs-decorated class.".format(o=object)
|
|
|
|
) == e.value.args[0]
|
|
|
|
|
|
|
|
@given(simple_classes())
|
|
|
|
def test_fields_dict(self, C):
|
|
|
|
"""
|
|
|
|
Returns an ordered dict of ``{attribute_name: Attribute}``.
|
|
|
|
"""
|
|
|
|
d = fields_dict(C)
|
|
|
|
|
|
|
|
assert isinstance(d, ordered_dict)
|
|
|
|
assert list(fields(C)) == list(d.values())
|
|
|
|
assert [a.name for a in fields(C)] == [field_name for field_name in d]
|
|
|
|
|
|
|
|
|
2017-12-23 07:46:10 +00:00
|
|
|
class TestConverter(object):
|
2015-09-16 15:45:51 +00:00
|
|
|
"""
|
|
|
|
Tests for attribute conversion.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2015-09-16 15:45:51 +00:00
|
|
|
def test_convert(self):
|
|
|
|
"""
|
2017-12-23 07:46:10 +00:00
|
|
|
Return value of converter is used as the attribute's value.
|
2015-09-16 15:45:51 +00:00
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
C = make_class(
|
|
|
|
"C", {"x": attr.ib(converter=lambda v: v + 1), "y": attr.ib()}
|
|
|
|
)
|
2015-09-16 15:45:51 +00:00
|
|
|
c = C(1, 2)
|
2017-10-02 17:32:10 +00:00
|
|
|
|
2015-09-16 15:45:51 +00:00
|
|
|
assert c.x == 2
|
|
|
|
assert c.y == 2
|
|
|
|
|
2016-09-10 06:23:21 +00:00
|
|
|
@given(integers(), booleans())
|
|
|
|
def test_convert_property(self, val, init):
|
|
|
|
"""
|
|
|
|
Property tests for attributes with convert.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
C = make_class(
|
|
|
|
"C",
|
|
|
|
{
|
|
|
|
"y": attr.ib(),
|
|
|
|
"x": attr.ib(
|
|
|
|
init=init, default=val, converter=lambda v: v + 1
|
|
|
|
),
|
|
|
|
},
|
|
|
|
)
|
2016-09-10 06:23:21 +00:00
|
|
|
c = C(2)
|
2017-10-02 17:32:10 +00:00
|
|
|
|
2016-09-10 06:23:21 +00:00
|
|
|
assert c.x == val + 1
|
|
|
|
assert c.y == 2
|
|
|
|
|
|
|
|
@given(integers(), booleans())
|
|
|
|
def test_convert_factory_property(self, val, init):
|
|
|
|
"""
|
|
|
|
Property tests for attributes with convert, and a factory default.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
C = make_class(
|
|
|
|
"C",
|
|
|
|
ordered_dict(
|
|
|
|
[
|
|
|
|
("y", attr.ib()),
|
|
|
|
(
|
|
|
|
"x",
|
|
|
|
attr.ib(
|
|
|
|
init=init,
|
|
|
|
default=Factory(lambda: val),
|
|
|
|
converter=lambda v: v + 1,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
]
|
|
|
|
),
|
|
|
|
)
|
2016-09-10 06:23:21 +00:00
|
|
|
c = C(2)
|
2017-10-02 17:32:10 +00:00
|
|
|
|
2016-09-10 06:23:21 +00:00
|
|
|
assert c.x == val + 1
|
|
|
|
assert c.y == 2
|
|
|
|
|
2017-05-16 07:36:39 +00:00
|
|
|
def test_factory_takes_self(self):
|
|
|
|
"""
|
|
|
|
If takes_self on factories is True, self is passed.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
C = make_class(
|
|
|
|
"C",
|
|
|
|
{
|
|
|
|
"x": attr.ib(
|
|
|
|
default=Factory((lambda self: self), takes_self=True)
|
|
|
|
)
|
|
|
|
},
|
|
|
|
)
|
2017-05-16 07:36:39 +00:00
|
|
|
|
|
|
|
i = C()
|
|
|
|
|
|
|
|
assert i is i.x
|
|
|
|
|
2017-06-06 11:13:41 +00:00
|
|
|
def test_factory_hashable(self):
|
|
|
|
"""
|
|
|
|
Factory is hashable.
|
|
|
|
"""
|
|
|
|
assert hash(Factory(None, False)) == hash(Factory(None, False))
|
|
|
|
|
2015-09-16 20:12:46 +00:00
|
|
|
def test_convert_before_validate(self):
|
2015-09-16 15:45:51 +00:00
|
|
|
"""
|
2015-09-16 20:12:46 +00:00
|
|
|
Validation happens after conversion.
|
2015-09-16 15:45:51 +00:00
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2015-09-16 15:45:51 +00:00
|
|
|
def validator(inst, attr, val):
|
|
|
|
raise RuntimeError("foo")
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2015-09-16 15:55:17 +00:00
|
|
|
C = make_class(
|
2018-06-10 17:40:07 +00:00
|
|
|
"C",
|
|
|
|
{
|
2017-12-23 07:46:10 +00:00
|
|
|
"x": attr.ib(validator=validator, converter=lambda v: 1 / 0),
|
2017-10-02 17:32:10 +00:00
|
|
|
"y": attr.ib(),
|
2018-06-10 17:40:07 +00:00
|
|
|
},
|
|
|
|
)
|
2015-09-16 20:12:46 +00:00
|
|
|
with pytest.raises(ZeroDivisionError):
|
2015-09-16 15:45:51 +00:00
|
|
|
C(1, 2)
|
|
|
|
|
2016-09-05 12:47:06 +00:00
|
|
|
def test_frozen(self):
|
|
|
|
"""
|
|
|
|
Converters circumvent immutability.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
C = make_class(
|
|
|
|
"C", {"x": attr.ib(converter=lambda v: int(v))}, frozen=True
|
|
|
|
)
|
2016-09-05 12:47:06 +00:00
|
|
|
C("1")
|
|
|
|
|
2017-12-23 07:46:10 +00:00
|
|
|
def test_deprecated_convert(self):
|
|
|
|
"""
|
|
|
|
Using *convert* raises a DeprecationWarning and sets the converter
|
|
|
|
field.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-12-23 07:46:10 +00:00
|
|
|
def conv(v):
|
|
|
|
return v
|
|
|
|
|
|
|
|
with pytest.warns(DeprecationWarning) as wi:
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-12-23 07:46:10 +00:00
|
|
|
@attr.s
|
|
|
|
class C(object):
|
|
|
|
x = attr.ib(convert=conv)
|
|
|
|
|
|
|
|
convert = fields(C).x.convert
|
|
|
|
|
|
|
|
assert 2 == len(wi.list)
|
|
|
|
w = wi.pop()
|
|
|
|
|
|
|
|
assert conv == fields(C).x.converter == convert
|
|
|
|
assert (
|
|
|
|
"The `convert` argument is deprecated in favor of `converter`. "
|
|
|
|
"It will be removed after 2019/01.",
|
|
|
|
) == w.message.args
|
|
|
|
assert __file__ == w.filename
|
|
|
|
|
|
|
|
def test_convert_converter(self):
|
|
|
|
"""
|
|
|
|
A TypeError is raised if both *convert* and *converter* are passed.
|
|
|
|
"""
|
|
|
|
with pytest.raises(RuntimeError) as ei:
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-12-23 07:46:10 +00:00
|
|
|
@attr.s
|
|
|
|
class C(object):
|
|
|
|
x = attr.ib(convert=lambda v: v, converter=lambda v: v)
|
|
|
|
|
|
|
|
assert (
|
|
|
|
"Can't pass both `convert` and `converter`. "
|
|
|
|
"Please use `converter` only.",
|
|
|
|
) == ei.value.args
|
|
|
|
|
2015-09-16 15:45:51 +00:00
|
|
|
|
2015-02-09 12:16:56 +00:00
|
|
|
class TestValidate(object):
|
|
|
|
"""
|
|
|
|
Tests for `validate`.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2015-02-09 12:16:56 +00:00
|
|
|
def test_success(self):
|
|
|
|
"""
|
2017-06-11 17:37:56 +00:00
|
|
|
If the validator succeeds, nothing gets raised.
|
2015-02-09 12:16:56 +00:00
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
C = make_class(
|
|
|
|
"C", {"x": attr.ib(validator=lambda *a: None), "y": attr.ib()}
|
|
|
|
)
|
2015-02-09 12:16:56 +00:00
|
|
|
validate(C(1, 2))
|
|
|
|
|
|
|
|
def test_propagates(self):
|
|
|
|
"""
|
|
|
|
The exception of the validator is handed through.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2015-02-09 12:16:56 +00:00
|
|
|
def raiser(_, __, value):
|
|
|
|
if value == 42:
|
|
|
|
raise FloatingPointError
|
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
C = make_class("C", {"x": attr.ib(validator=raiser)})
|
2015-02-09 12:16:56 +00:00
|
|
|
i = C(1)
|
|
|
|
i.x = 42
|
|
|
|
|
|
|
|
with pytest.raises(FloatingPointError):
|
|
|
|
validate(i)
|
2015-02-20 10:30:46 +00:00
|
|
|
|
|
|
|
def test_run_validators(self):
|
|
|
|
"""
|
|
|
|
Setting `_run_validators` to False prevents validators from running.
|
|
|
|
"""
|
|
|
|
_config._run_validators = False
|
|
|
|
obj = object()
|
|
|
|
|
|
|
|
def raiser(_, __, ___):
|
|
|
|
raise Exception(obj)
|
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
C = make_class("C", {"x": attr.ib(validator=raiser)})
|
2016-08-22 14:35:08 +00:00
|
|
|
c = C(1)
|
|
|
|
validate(c)
|
|
|
|
assert 1 == c.x
|
2015-02-20 10:30:46 +00:00
|
|
|
_config._run_validators = True
|
|
|
|
|
2016-08-22 14:35:08 +00:00
|
|
|
with pytest.raises(Exception):
|
|
|
|
validate(c)
|
|
|
|
|
2015-02-20 10:30:46 +00:00
|
|
|
with pytest.raises(Exception) as e:
|
|
|
|
C(1)
|
|
|
|
assert (obj,) == e.value.args
|
2016-11-19 08:47:03 +00:00
|
|
|
|
2017-02-04 13:37:49 +00:00
|
|
|
def test_multiple_validators(self):
|
|
|
|
"""
|
|
|
|
If a list is passed as a validator, all of its items are treated as one
|
|
|
|
and must pass.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-02-04 13:37:49 +00:00
|
|
|
def v1(_, __, value):
|
|
|
|
if value == 23:
|
|
|
|
raise TypeError("omg")
|
|
|
|
|
|
|
|
def v2(_, __, value):
|
|
|
|
if value == 42:
|
|
|
|
raise ValueError("omg")
|
|
|
|
|
2017-10-02 17:32:10 +00:00
|
|
|
C = make_class("C", {"x": attr.ib(validator=[v1, v2])})
|
2017-02-04 13:37:49 +00:00
|
|
|
|
|
|
|
validate(C(1))
|
|
|
|
|
|
|
|
with pytest.raises(TypeError) as e:
|
|
|
|
C(23)
|
|
|
|
|
|
|
|
assert "omg" == e.value.args[0]
|
|
|
|
|
|
|
|
with pytest.raises(ValueError) as e:
|
|
|
|
C(42)
|
|
|
|
|
|
|
|
assert "omg" == e.value.args[0]
|
|
|
|
|
|
|
|
def test_multiple_empty(self):
|
|
|
|
"""
|
|
|
|
Empty list/tuple for validator is the same as None.
|
|
|
|
"""
|
2017-10-02 17:32:10 +00:00
|
|
|
C1 = make_class("C", {"x": attr.ib(validator=[])})
|
|
|
|
C2 = make_class("C", {"x": attr.ib(validator=None)})
|
2017-02-04 13:37:49 +00:00
|
|
|
|
|
|
|
assert inspect.getsource(C1.__init__) == inspect.getsource(C2.__init__)
|
|
|
|
|
2016-11-19 08:47:03 +00:00
|
|
|
|
|
|
|
# Hypothesis seems to cache values, so the lists of attributes come out
|
|
|
|
# unsorted.
|
|
|
|
sorted_lists_of_attrs = list_of_attrs.map(
|
2018-06-10 17:40:07 +00:00
|
|
|
lambda l: sorted(l, key=attrgetter("counter"))
|
|
|
|
)
|
2016-11-19 08:47:03 +00:00
|
|
|
|
|
|
|
|
|
|
|
class TestMetadata(object):
|
|
|
|
"""
|
|
|
|
Tests for metadata handling.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2016-11-19 08:47:03 +00:00
|
|
|
@given(sorted_lists_of_attrs)
|
|
|
|
def test_metadata_present(self, list_of_attrs):
|
|
|
|
"""
|
|
|
|
Assert dictionaries are copied and present.
|
|
|
|
"""
|
|
|
|
C = make_class("C", dict(zip(gen_attr_names(), list_of_attrs)))
|
|
|
|
|
|
|
|
for hyp_attr, class_attr in zip(list_of_attrs, fields(C)):
|
|
|
|
if hyp_attr.metadata is None:
|
|
|
|
# The default is a singleton empty dict.
|
|
|
|
assert class_attr.metadata is not None
|
|
|
|
assert len(class_attr.metadata) == 0
|
|
|
|
else:
|
|
|
|
assert hyp_attr.metadata == class_attr.metadata
|
|
|
|
|
|
|
|
# Once more, just to assert getting items and iteration.
|
|
|
|
for k in class_attr.metadata:
|
|
|
|
assert hyp_attr.metadata[k] == class_attr.metadata[k]
|
2018-06-10 17:40:07 +00:00
|
|
|
assert hyp_attr.metadata.get(k) == class_attr.metadata.get(
|
|
|
|
k
|
|
|
|
)
|
2016-11-19 08:47:03 +00:00
|
|
|
|
|
|
|
@given(simple_classes(), text())
|
|
|
|
def test_metadata_immutability(self, C, string):
|
|
|
|
"""
|
|
|
|
The metadata dict should be best-effort immutable.
|
|
|
|
"""
|
|
|
|
for a in fields(C):
|
|
|
|
with pytest.raises(TypeError):
|
|
|
|
a.metadata[string] = string
|
|
|
|
with pytest.raises(AttributeError):
|
|
|
|
a.metadata.update({string: string})
|
|
|
|
with pytest.raises(AttributeError):
|
|
|
|
a.metadata.clear()
|
|
|
|
with pytest.raises(AttributeError):
|
|
|
|
a.metadata.setdefault(string, string)
|
|
|
|
|
|
|
|
for k in a.metadata:
|
|
|
|
# For some reason, Python 3's MappingProxyType throws an
|
|
|
|
# IndexError for deletes on a large integer key.
|
|
|
|
with pytest.raises((TypeError, IndexError)):
|
|
|
|
del a.metadata[k]
|
|
|
|
with pytest.raises(AttributeError):
|
|
|
|
a.metadata.pop(k)
|
|
|
|
with pytest.raises(AttributeError):
|
2018-06-10 17:40:07 +00:00
|
|
|
a.metadata.popitem()
|
2016-11-19 08:47:03 +00:00
|
|
|
|
|
|
|
@given(lists(simple_attrs_without_metadata, min_size=2, max_size=5))
|
|
|
|
def test_empty_metadata_singleton(self, list_of_attrs):
|
|
|
|
"""
|
|
|
|
All empty metadata attributes share the same empty metadata dict.
|
|
|
|
"""
|
|
|
|
C = make_class("C", dict(zip(gen_attr_names(), list_of_attrs)))
|
|
|
|
for a in fields(C)[1:]:
|
|
|
|
assert a.metadata is fields(C)[0].metadata
|
2017-10-26 10:55:34 +00:00
|
|
|
|
2017-12-19 06:51:24 +00:00
|
|
|
@given(lists(simple_attrs_without_metadata, min_size=2, max_size=5))
|
|
|
|
def test_empty_countingattr_metadata_independent(self, list_of_attrs):
|
|
|
|
"""
|
|
|
|
All empty metadata attributes are independent before ``@attr.s``.
|
|
|
|
"""
|
|
|
|
for x, y in itertools.combinations(list_of_attrs, 2):
|
|
|
|
assert x.metadata is not y.metadata
|
|
|
|
|
|
|
|
@given(lists(simple_attrs_with_metadata(), min_size=2, max_size=5))
|
|
|
|
def test_not_none_metadata(self, list_of_attrs):
|
|
|
|
"""
|
|
|
|
Non-empty metadata attributes exist as fields after ``@attr.s``.
|
|
|
|
"""
|
|
|
|
C = make_class("C", dict(zip(gen_attr_names(), list_of_attrs)))
|
|
|
|
|
|
|
|
assert len(fields(C)) > 0
|
|
|
|
|
|
|
|
for cls_a, raw_a in zip(fields(C), list_of_attrs):
|
|
|
|
assert cls_a.metadata != {}
|
|
|
|
assert cls_a.metadata == raw_a.metadata
|
|
|
|
|
2017-12-20 11:43:22 +00:00
|
|
|
def test_metadata(self):
|
2017-12-19 06:51:24 +00:00
|
|
|
"""
|
2017-12-20 11:43:22 +00:00
|
|
|
If metadata that is not None is passed, it is used.
|
|
|
|
|
|
|
|
This is necessary for coverage because the previous test is
|
|
|
|
hypothesis-based.
|
2017-12-19 06:51:24 +00:00
|
|
|
"""
|
2017-12-20 11:43:22 +00:00
|
|
|
md = {}
|
|
|
|
a = attr.ib(metadata=md)
|
|
|
|
|
|
|
|
assert md is a.metadata
|
2017-12-19 06:51:24 +00:00
|
|
|
|
2017-10-26 10:55:34 +00:00
|
|
|
|
|
|
|
class TestClassBuilder(object):
|
|
|
|
"""
|
|
|
|
Tests for `_ClassBuilder`.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-26 10:55:34 +00:00
|
|
|
def test_repr_str(self):
|
|
|
|
"""
|
|
|
|
Trying to add a `__str__` without having a `__repr__` raises a
|
|
|
|
ValueError.
|
|
|
|
"""
|
|
|
|
with pytest.raises(ValueError) as ei:
|
|
|
|
make_class("C", {}, repr=False, str=True)
|
|
|
|
|
|
|
|
assert (
|
|
|
|
"__str__ can only be generated if a __repr__ exists.",
|
|
|
|
) == ei.value.args
|
|
|
|
|
|
|
|
def test_repr(self):
|
|
|
|
"""
|
|
|
|
repr of builder itself makes sense.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-26 10:55:34 +00:00
|
|
|
class C(object):
|
|
|
|
pass
|
|
|
|
|
2017-11-08 10:15:21 +00:00
|
|
|
b = _ClassBuilder(C, None, True, True, False)
|
2017-10-26 10:55:34 +00:00
|
|
|
|
|
|
|
assert "<_ClassBuilder(cls=C)>" == repr(b)
|
|
|
|
|
|
|
|
def test_returns_self(self):
|
|
|
|
"""
|
|
|
|
All methods return the builder for chaining.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-10-26 10:55:34 +00:00
|
|
|
class C(object):
|
|
|
|
x = attr.ib()
|
|
|
|
|
2017-11-08 10:15:21 +00:00
|
|
|
b = _ClassBuilder(C, None, True, True, False)
|
2017-10-26 10:55:34 +00:00
|
|
|
|
2018-06-10 17:40:07 +00:00
|
|
|
cls = (
|
|
|
|
b.add_cmp()
|
|
|
|
.add_hash()
|
|
|
|
.add_init()
|
|
|
|
.add_repr("ns")
|
|
|
|
.add_str()
|
2017-10-26 10:55:34 +00:00
|
|
|
.build_class()
|
2018-06-10 17:40:07 +00:00
|
|
|
)
|
2017-10-26 10:55:34 +00:00
|
|
|
|
|
|
|
assert "ns.C(x=1)" == repr(cls(1))
|
2017-12-27 10:54:18 +00:00
|
|
|
|
2018-06-10 17:40:07 +00:00
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"meth_name",
|
|
|
|
[
|
|
|
|
"__init__",
|
|
|
|
"__hash__",
|
|
|
|
"__repr__",
|
|
|
|
"__str__",
|
|
|
|
"__eq__",
|
|
|
|
"__ne__",
|
|
|
|
"__lt__",
|
|
|
|
"__le__",
|
|
|
|
"__gt__",
|
|
|
|
"__ge__",
|
|
|
|
],
|
|
|
|
)
|
2017-12-27 10:54:18 +00:00
|
|
|
def test_attaches_meta_dunders(self, meth_name):
|
|
|
|
"""
|
|
|
|
Generated methods have correct __module__, __name__, and __qualname__
|
|
|
|
attributes.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-12-27 10:54:18 +00:00
|
|
|
@attr.s(hash=True, str=True)
|
|
|
|
class C(object):
|
|
|
|
def organic(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
meth = getattr(C, meth_name)
|
|
|
|
|
|
|
|
assert meth_name == meth.__name__
|
|
|
|
assert C.organic.__module__ == meth.__module__
|
|
|
|
if not PY2:
|
|
|
|
organic_prefix = C.organic.__qualname__.rsplit(".", 1)[0]
|
|
|
|
assert organic_prefix + "." + meth_name == meth.__qualname__
|
|
|
|
|
|
|
|
def test_handles_missing_meta_on_class(self):
|
|
|
|
"""
|
|
|
|
If the class hasn't a __module__ or __qualname__, the method hasn't
|
|
|
|
either.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2017-12-27 10:54:18 +00:00
|
|
|
class C(object):
|
|
|
|
pass
|
|
|
|
|
|
|
|
b = _ClassBuilder(
|
2018-06-10 17:40:07 +00:00
|
|
|
C, these=None, slots=False, frozen=False, auto_attribs=False
|
2017-12-27 10:54:18 +00:00
|
|
|
)
|
|
|
|
b._cls = {} # no __module__; no __qualname__
|
|
|
|
|
|
|
|
def fake_meth(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
fake_meth.__module__ = "42"
|
|
|
|
fake_meth.__qualname__ = "23"
|
|
|
|
|
|
|
|
rv = b._add_method_dunders(fake_meth)
|
|
|
|
|
|
|
|
assert "42" == rv.__module__ == fake_meth.__module__
|
|
|
|
assert "23" == rv.__qualname__ == fake_meth.__qualname__
|
2018-01-17 12:16:39 +00:00
|
|
|
|
|
|
|
def test_weakref_setstate(self):
|
|
|
|
"""
|
|
|
|
__weakref__ is not set on in setstate because it's not writable in
|
|
|
|
slots classes.
|
|
|
|
"""
|
2018-06-10 17:40:07 +00:00
|
|
|
|
2018-01-17 12:16:39 +00:00
|
|
|
@attr.s(slots=True)
|
|
|
|
class C(object):
|
|
|
|
__weakref__ = attr.ib(
|
|
|
|
init=False, hash=False, repr=False, cmp=False
|
|
|
|
)
|
|
|
|
|
|
|
|
assert C() == copy.deepcopy(C())
|