test_modules.py 4.11 KB
Newer Older
1
2
3
import pytest

import env
4
from pybind11_tests import ConstructorStats
5
6
7
from pybind11_tests import modules as m
from pybind11_tests.modules import subsubmodule as ms

Dean Moldovan's avatar
Dean Moldovan committed
8

9
def test_nested_modules():
Dean Moldovan's avatar
Dean Moldovan committed
10
    import pybind11_tests
11

Dean Moldovan's avatar
Dean Moldovan committed
12
    assert pybind11_tests.__name__ == "pybind11_tests"
13
    assert pybind11_tests.modules.__name__ == "pybind11_tests.modules"
14
15
16
17
    assert (
        pybind11_tests.modules.subsubmodule.__name__
        == "pybind11_tests.modules.subsubmodule"
    )
18
19
    assert m.__name__ == "pybind11_tests.modules"
    assert ms.__name__ == "pybind11_tests.modules.subsubmodule"
Dean Moldovan's avatar
Dean Moldovan committed
20

21
    assert ms.submodule_func() == "submodule_func()"
Dean Moldovan's avatar
Dean Moldovan committed
22
23
24


def test_reference_internal():
25
    b = ms.B()
Dean Moldovan's avatar
Dean Moldovan committed
26
27
28
29
30
    assert str(b.get_a1()) == "A[1]"
    assert str(b.a1) == "A[1]"
    assert str(b.get_a2()) == "A[2]"
    assert str(b.a2) == "A[2]"

31
32
    b.a1 = ms.A(42)
    b.a2 = ms.A(43)
Dean Moldovan's avatar
Dean Moldovan committed
33
34
35
36
37
    assert str(b.get_a1()) == "A[42]"
    assert str(b.a1) == "A[42]"
    assert str(b.get_a2()) == "A[43]"
    assert str(b.a2) == "A[43]"

38
    astats, bstats = ConstructorStats.get(ms.A), ConstructorStats.get(ms.B)
Dean Moldovan's avatar
Dean Moldovan committed
39
40
41
42
43
    assert astats.alive() == 2
    assert bstats.alive() == 1
    del b
    assert astats.alive() == 0
    assert bstats.alive() == 0
44
    assert astats.values() == ["1", "2", "42", "43"]
Dean Moldovan's avatar
Dean Moldovan committed
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
    assert bstats.values() == []
    assert astats.default_constructions == 0
    assert bstats.default_constructions == 1
    assert astats.copy_constructions == 0
    assert bstats.copy_constructions == 0
    # assert astats.move_constructions >= 0  # Don't invoke any
    # assert bstats.move_constructions >= 0  # Don't invoke any
    assert astats.copy_assignments == 2
    assert bstats.copy_assignments == 0
    assert astats.move_assignments == 0
    assert bstats.move_assignments == 0


def test_importing():
    from collections import OrderedDict

61
62
    from pybind11_tests.modules import OD

Dean Moldovan's avatar
Dean Moldovan committed
63
    assert OD is OrderedDict
64
    assert str(OD([(1, "a"), (2, "b")])) == "OrderedDict([(1, 'a'), (2, 'b')])"
65
66
67
68
69
70


def test_pydoc():
    """Pydoc needs to be able to provide help() for everything inside a pybind11 module"""
    import pydoc

71
72
    import pybind11_tests

73
    assert pybind11_tests.__name__ == "pybind11_tests"
74
    assert pybind11_tests.__doc__ == "pybind11 test module"
75
    assert pydoc.text.docmodule(pybind11_tests)
76
77
78
79
80


def test_duplicate_registration():
    """Registering two things with the same name"""

81
    assert m.duplicate_registration() == []
82
83
84
85
86
87
88
89
90
91
92
93
94


def test_builtin_key_type():
    """Test that all the keys in the builtin modules have type str.

    Previous versions of pybind11 would add a unicode key in python 2.
    """
    if hasattr(__builtins__, "keys"):
        keys = __builtins__.keys()
    else:  # this is to make pypy happy since builtins is different there.
        keys = __builtins__.__dict__.keys()

    assert {type(k) for k in keys} == {str}
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109


@pytest.mark.xfail("env.PYPY", reason="PyModule_GetName()")
def test_def_submodule_failures():
    sm = m.def_submodule(m, b"ScratchSubModuleName")  # Using bytes to show it works.
    assert sm.__name__ == m.__name__ + "." + "ScratchSubModuleName"
    malformed_utf8 = b"\x80"
    if env.PYPY:
        # It is not worth the effort finding a trigger for a failure when running with PyPy.
        pytest.skip("Sufficiently exercised on platforms other than PyPy.")
    else:
        # Meant to trigger PyModule_GetName() failure:
        sm_name_orig = sm.__name__
        sm.__name__ = malformed_utf8
        try:
110
111
112
113
            # We want to assert that a bad __name__ causes some kind of failure, although we do not want to exercise
            # the internals of PyModule_GetName(). Currently all supported Python versions raise SystemError. If that
            # changes in future Python versions, simply add the new expected exception types here.
            with pytest.raises(SystemError):
114
115
116
117
118
119
120
                m.def_submodule(sm, b"SubSubModuleName")
        finally:
            # Clean up to ensure nothing gets upset by a module with an invalid __name__.
            sm.__name__ = sm_name_orig  # Purely precautionary.
    # Meant to trigger PyImport_AddModule() failure:
    with pytest.raises(UnicodeDecodeError):
        m.def_submodule(sm, malformed_utf8)