mirror of
https://github.com/RYDE-WORK/pybind11.git
synced 2026-01-19 21:23:26 +08:00
* Change Python version guard: PYTHON < 3.7 IS UNSUPPORTED. * Replace or remove Python 3.6 jobs. * Move appveyor to Python 3.8 * Change `[tool.pylint]` `master.py-version` from `3.6` to `3.8` * Change `[tool.pylint]` `master.py-version` to `3.7` * Remove `centos:7` job; Change almalinux:8 job to use Python 3.8 * Try 🐍 3.8 • ubuntu-20.04 • x64 without `-DCMAKE_CXX_FLAGS="-D_=1"` * Update setup.cfg as suggested by @henryiii * Try running `cmake --build . --target cpptest` on all platforms (`standard` job). * Disable deadsnakes jobs entirely. * Apply PR #5179: Add Python 3.10, 3.11, 3.12 to win32 job matrix. * Add back `-DCMAKE_CXX_FLAGS="-D_=1"` but do not install boost in that case. * PY_VERSION_HEX < 3.7 cleanup pass: include/pybind11 * WITH_THREAD cleanup pass: include/pybind11 * Undo incorrect change. * Revert "Disable deadsnakes jobs entirely." This reverts commit bbcd0087b2d52e0130f96792dd5dd03704280a57. * WITH_THREAD cleanup pass: tests/ * Change Python version guard in pybind11/__init__.py: pybind11 does not support Python < 3.7. * Misc cleanup pass * chore: use future imports Signed-off-by: Henry Schreiner <henryschreineriii@gmail.com> * Update tests/test_numpy_array.py * Update test_numpy_array.py --------- Signed-off-by: Henry Schreiner <henryschreineriii@gmail.com> Co-authored-by: Henry Schreiner <henryschreineriii@gmail.com>
250 lines
6.4 KiB
Python
250 lines
6.4 KiB
Python
from __future__ import annotations
|
|
|
|
import pytest
|
|
|
|
import env # noqa: F401
|
|
from pybind11_tests import ConstructorStats
|
|
from pybind11_tests import call_policies as m
|
|
|
|
|
|
@pytest.mark.xfail("env.PYPY", reason="sometimes comes out 1 off on PyPy", strict=False)
|
|
def test_keep_alive_argument(capture):
|
|
n_inst = ConstructorStats.detail_reg_inst()
|
|
with capture:
|
|
p = m.Parent()
|
|
assert capture == "Allocating parent."
|
|
with capture:
|
|
p.addChild(m.Child())
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 1
|
|
assert (
|
|
capture
|
|
== """
|
|
Allocating child.
|
|
Releasing child.
|
|
"""
|
|
)
|
|
with capture:
|
|
del p
|
|
assert ConstructorStats.detail_reg_inst() == n_inst
|
|
assert capture == "Releasing parent."
|
|
|
|
with capture:
|
|
p = m.Parent()
|
|
assert capture == "Allocating parent."
|
|
with capture:
|
|
p.addChildKeepAlive(m.Child())
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 2
|
|
assert capture == "Allocating child."
|
|
with capture:
|
|
del p
|
|
assert ConstructorStats.detail_reg_inst() == n_inst
|
|
assert (
|
|
capture
|
|
== """
|
|
Releasing parent.
|
|
Releasing child.
|
|
"""
|
|
)
|
|
|
|
p = m.Parent()
|
|
c = m.Child()
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 2
|
|
m.free_function(p, c)
|
|
del c
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 2
|
|
del p
|
|
assert ConstructorStats.detail_reg_inst() == n_inst
|
|
|
|
with pytest.raises(RuntimeError) as excinfo:
|
|
m.invalid_arg_index()
|
|
assert str(excinfo.value) == "Could not activate keep_alive!"
|
|
|
|
|
|
def test_keep_alive_return_value(capture):
|
|
n_inst = ConstructorStats.detail_reg_inst()
|
|
with capture:
|
|
p = m.Parent()
|
|
assert capture == "Allocating parent."
|
|
with capture:
|
|
p.returnChild()
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 1
|
|
assert (
|
|
capture
|
|
== """
|
|
Allocating child.
|
|
Releasing child.
|
|
"""
|
|
)
|
|
with capture:
|
|
del p
|
|
assert ConstructorStats.detail_reg_inst() == n_inst
|
|
assert capture == "Releasing parent."
|
|
|
|
with capture:
|
|
p = m.Parent()
|
|
assert capture == "Allocating parent."
|
|
with capture:
|
|
p.returnChildKeepAlive()
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 2
|
|
assert capture == "Allocating child."
|
|
with capture:
|
|
del p
|
|
assert ConstructorStats.detail_reg_inst() == n_inst
|
|
assert (
|
|
capture
|
|
== """
|
|
Releasing parent.
|
|
Releasing child.
|
|
"""
|
|
)
|
|
|
|
p = m.Parent()
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 1
|
|
with capture:
|
|
m.Parent.staticFunction(p)
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 2
|
|
assert capture == "Allocating child."
|
|
with capture:
|
|
del p
|
|
assert ConstructorStats.detail_reg_inst() == n_inst
|
|
assert (
|
|
capture
|
|
== """
|
|
Releasing parent.
|
|
Releasing child.
|
|
"""
|
|
)
|
|
|
|
|
|
# https://foss.heptapod.net/pypy/pypy/-/issues/2447
|
|
@pytest.mark.xfail("env.PYPY", reason="_PyObject_GetDictPtr is unimplemented")
|
|
def test_alive_gc(capture):
|
|
n_inst = ConstructorStats.detail_reg_inst()
|
|
p = m.ParentGC()
|
|
p.addChildKeepAlive(m.Child())
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 2
|
|
lst = [p]
|
|
lst.append(lst) # creates a circular reference
|
|
with capture:
|
|
del p, lst
|
|
assert ConstructorStats.detail_reg_inst() == n_inst
|
|
assert (
|
|
capture
|
|
== """
|
|
Releasing parent.
|
|
Releasing child.
|
|
"""
|
|
)
|
|
|
|
|
|
def test_alive_gc_derived(capture):
|
|
class Derived(m.Parent):
|
|
pass
|
|
|
|
n_inst = ConstructorStats.detail_reg_inst()
|
|
p = Derived()
|
|
p.addChildKeepAlive(m.Child())
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 2
|
|
lst = [p]
|
|
lst.append(lst) # creates a circular reference
|
|
with capture:
|
|
del p, lst
|
|
assert ConstructorStats.detail_reg_inst() == n_inst
|
|
assert (
|
|
capture
|
|
== """
|
|
Releasing parent.
|
|
Releasing child.
|
|
"""
|
|
)
|
|
|
|
|
|
def test_alive_gc_multi_derived(capture):
|
|
class Derived(m.Parent, m.Child):
|
|
def __init__(self):
|
|
m.Parent.__init__(self)
|
|
m.Child.__init__(self)
|
|
|
|
n_inst = ConstructorStats.detail_reg_inst()
|
|
p = Derived()
|
|
p.addChildKeepAlive(m.Child())
|
|
# +3 rather than +2 because Derived corresponds to two registered instances
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 3
|
|
lst = [p]
|
|
lst.append(lst) # creates a circular reference
|
|
with capture:
|
|
del p, lst
|
|
assert ConstructorStats.detail_reg_inst() == n_inst
|
|
assert (
|
|
capture
|
|
== """
|
|
Releasing parent.
|
|
Releasing child.
|
|
Releasing child.
|
|
"""
|
|
)
|
|
|
|
|
|
def test_return_none(capture):
|
|
n_inst = ConstructorStats.detail_reg_inst()
|
|
with capture:
|
|
p = m.Parent()
|
|
assert capture == "Allocating parent."
|
|
with capture:
|
|
p.returnNullChildKeepAliveChild()
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 1
|
|
assert capture == ""
|
|
with capture:
|
|
del p
|
|
assert ConstructorStats.detail_reg_inst() == n_inst
|
|
assert capture == "Releasing parent."
|
|
|
|
with capture:
|
|
p = m.Parent()
|
|
assert capture == "Allocating parent."
|
|
with capture:
|
|
p.returnNullChildKeepAliveParent()
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 1
|
|
assert capture == ""
|
|
with capture:
|
|
del p
|
|
assert ConstructorStats.detail_reg_inst() == n_inst
|
|
assert capture == "Releasing parent."
|
|
|
|
|
|
def test_keep_alive_constructor(capture):
|
|
n_inst = ConstructorStats.detail_reg_inst()
|
|
|
|
with capture:
|
|
p = m.Parent(m.Child())
|
|
assert ConstructorStats.detail_reg_inst() == n_inst + 2
|
|
assert (
|
|
capture
|
|
== """
|
|
Allocating child.
|
|
Allocating parent.
|
|
"""
|
|
)
|
|
with capture:
|
|
del p
|
|
assert ConstructorStats.detail_reg_inst() == n_inst
|
|
assert (
|
|
capture
|
|
== """
|
|
Releasing parent.
|
|
Releasing child.
|
|
"""
|
|
)
|
|
|
|
|
|
def test_call_guard():
|
|
assert m.unguarded_call() == "unguarded"
|
|
assert m.guarded_call() == "guarded"
|
|
|
|
assert m.multiple_guards_correct_order() == "guarded & guarded"
|
|
assert m.multiple_guards_wrong_order() == "unguarded & guarded"
|
|
|
|
if hasattr(m, "with_gil"):
|
|
assert m.with_gil() == "GIL held"
|
|
assert m.without_gil() == "GIL released"
|