117 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Python
		
	
	
		
		
			
		
	
	
			117 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Python
		
	
	
| 
								 | 
							
								import builtins
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								import pytest
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								import env
							 | 
						||
| 
								 | 
							
								from pybind11_tests import ConstructorStats
							 | 
						||
| 
								 | 
							
								from pybind11_tests import modules as m
							 | 
						||
| 
								 | 
							
								from pybind11_tests.modules import subsubmodule as ms
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def test_nested_modules():
							 | 
						||
| 
								 | 
							
								    import pybind11_tests
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    assert pybind11_tests.__name__ == "pybind11_tests"
							 | 
						||
| 
								 | 
							
								    assert pybind11_tests.modules.__name__ == "pybind11_tests.modules"
							 | 
						||
| 
								 | 
							
								    assert (
							 | 
						||
| 
								 | 
							
								        pybind11_tests.modules.subsubmodule.__name__
							 | 
						||
| 
								 | 
							
								        == "pybind11_tests.modules.subsubmodule"
							 | 
						||
| 
								 | 
							
								    )
							 | 
						||
| 
								 | 
							
								    assert m.__name__ == "pybind11_tests.modules"
							 | 
						||
| 
								 | 
							
								    assert ms.__name__ == "pybind11_tests.modules.subsubmodule"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    assert ms.submodule_func() == "submodule_func()"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def test_reference_internal():
							 | 
						||
| 
								 | 
							
								    b = ms.B()
							 | 
						||
| 
								 | 
							
								    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]"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    b.a1 = ms.A(42)
							 | 
						||
| 
								 | 
							
								    b.a2 = ms.A(43)
							 | 
						||
| 
								 | 
							
								    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]"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    astats, bstats = ConstructorStats.get(ms.A), ConstructorStats.get(ms.B)
							 | 
						||
| 
								 | 
							
								    assert astats.alive() == 2
							 | 
						||
| 
								 | 
							
								    assert bstats.alive() == 1
							 | 
						||
| 
								 | 
							
								    del b
							 | 
						||
| 
								 | 
							
								    assert astats.alive() == 0
							 | 
						||
| 
								 | 
							
								    assert bstats.alive() == 0
							 | 
						||
| 
								 | 
							
								    assert astats.values() == ["1", "2", "42", "43"]
							 | 
						||
| 
								 | 
							
								    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
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    from pybind11_tests.modules import OD
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    assert OD is OrderedDict
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def test_pydoc():
							 | 
						||
| 
								 | 
							
								    """Pydoc needs to be able to provide help() for everything inside a pybind11 module"""
							 | 
						||
| 
								 | 
							
								    import pydoc
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    import pybind11_tests
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    assert pybind11_tests.__name__ == "pybind11_tests"
							 | 
						||
| 
								 | 
							
								    assert pybind11_tests.__doc__ == "pybind11 test module"
							 | 
						||
| 
								 | 
							
								    assert pydoc.text.docmodule(pybind11_tests)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								def test_duplicate_registration():
							 | 
						||
| 
								 | 
							
								    """Registering two things with the same name"""
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    assert m.duplicate_registration() == []
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								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.
							 | 
						||
| 
								 | 
							
								    """
							 | 
						||
| 
								 | 
							
								    assert all(type(k) == str for k in dir(builtins))
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								@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:
							 | 
						||
| 
								 | 
							
								            # 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):
							 | 
						||
| 
								 | 
							
								                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)
							 |