"vscode:/vscode.git/clone" did not exist on "edbdef7ccf88df9632ad9e31884da347d1d3cd7c"
test_operator_overloading.py 3.23 KB
Newer Older
1
import pytest
2
from pybind11_tests import operators as m
3
4
5
from pybind11_tests import ConstructorStats


Dean Moldovan's avatar
Dean Moldovan committed
6
def test_operator_overloading():
7
8
    v1 = m.Vector2(1, 2)
    v2 = m.Vector(3, -1)
Dean Moldovan's avatar
Dean Moldovan committed
9
10
11
12
13
14
15
16
17
18
19
20
21
    assert str(v1) == "[1.000000, 2.000000]"
    assert str(v2) == "[3.000000, -1.000000]"

    assert str(v1 + v2) == "[4.000000, 1.000000]"
    assert str(v1 - v2) == "[-2.000000, 3.000000]"
    assert str(v1 - 8) == "[-7.000000, -6.000000]"
    assert str(v1 + 8) == "[9.000000, 10.000000]"
    assert str(v1 * 8) == "[8.000000, 16.000000]"
    assert str(v1 / 8) == "[0.125000, 0.250000]"
    assert str(8 - v1) == "[7.000000, 6.000000]"
    assert str(8 + v1) == "[9.000000, 10.000000]"
    assert str(8 * v1) == "[8.000000, 16.000000]"
    assert str(8 / v1) == "[8.000000, 4.000000]"
22
23
    assert str(v1 * v2) == "[3.000000, -2.000000]"
    assert str(v2 / v1) == "[3.000000, -0.500000]"
Dean Moldovan's avatar
Dean Moldovan committed
24

25
26
27
28
    v1 += 2 * v2
    assert str(v1) == "[7.000000, 0.000000]"
    v1 -= v2
    assert str(v1) == "[4.000000, 1.000000]"
Dean Moldovan's avatar
Dean Moldovan committed
29
30
    v1 *= 2
    assert str(v1) == "[8.000000, 2.000000]"
31
32
33
34
35
36
    v1 /= 16
    assert str(v1) == "[0.500000, 0.125000]"
    v1 *= v2
    assert str(v1) == "[1.500000, -0.125000]"
    v2 /= v1
    assert str(v2) == "[2.000000, 8.000000]"
Dean Moldovan's avatar
Dean Moldovan committed
37

38
    cstats = ConstructorStats.get(m.Vector2)
Dean Moldovan's avatar
Dean Moldovan committed
39
40
41
42
43
44
45
46
47
48
    assert cstats.alive() == 2
    del v1
    assert cstats.alive() == 1
    del v2
    assert cstats.alive() == 0
    assert cstats.values() == ['[1.000000, 2.000000]', '[3.000000, -1.000000]',
                               '[4.000000, 1.000000]', '[-2.000000, 3.000000]',
                               '[-7.000000, -6.000000]', '[9.000000, 10.000000]',
                               '[8.000000, 16.000000]', '[0.125000, 0.250000]',
                               '[7.000000, 6.000000]', '[9.000000, 10.000000]',
49
50
51
                               '[8.000000, 16.000000]', '[8.000000, 4.000000]',
                               '[3.000000, -2.000000]', '[3.000000, -0.500000]',
                               '[6.000000, -2.000000]']
Dean Moldovan's avatar
Dean Moldovan committed
52
53
54
55
56
    assert cstats.default_constructions == 0
    assert cstats.copy_constructions == 0
    assert cstats.move_constructions >= 10
    assert cstats.copy_assignments == 0
    assert cstats.move_assignments == 0
57
58
59
60
61


def test_operators_notimplemented():
    """#393: need to return NotSupported to ensure correct arithmetic operator behavior"""

62
    c1, c2 = m.C1(), m.C2()
63
64
65
66
67
68
69
70
71
    assert c1 + c1 == 11
    assert c2 + c2 == 22
    assert c2 + c1 == 21
    assert c1 + c2 == 12


def test_nested():
    """#328: first member in a class can't be used in operators"""

72
73
74
    a = m.NestA()
    b = m.NestB()
    c = m.NestC()
75
76

    a += 10
77
    assert m.get_NestA(a) == 13
78
    b.a += 100
79
    assert m.get_NestA(b.a) == 103
80
    c.b.a += 1000
81
    assert m.get_NestA(c.b.a) == 1003
82
    b -= 1
83
    assert m.get_NestB(b) == 3
84
    c.b -= 3
85
    assert m.get_NestB(c.b) == 1
86
    c *= 7
87
    assert m.get_NestC(c) == 35
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104

    abase = a.as_base()
    assert abase.value == -2
    a.as_base().value += 44
    assert abase.value == 42
    assert c.b.a.as_base().value == -2
    c.b.a.as_base().value += 44
    assert c.b.a.as_base().value == 42

    del c
    pytest.gc_collect()
    del a  # Should't delete while abase is still alive
    pytest.gc_collect()

    assert abase.value == 42
    del abase, b
    pytest.gc_collect()