Merge branch 'master' into smart_holder

This commit is contained in:
Ralf W. Grosse-Kunstleve 2023-08-07 20:59:51 -07:00
commit 6f702e7f0e
18 changed files with 145 additions and 89 deletions

View File

@ -405,54 +405,55 @@ jobs:
# run: cmake --build build --target test_cmake_build # run: cmake --build build --target test_cmake_build
# Testing on CentOS 7 + PGI compilers, which seems to require more workarounds # Testing on Ubuntu + NVHPC (previous PGI) compilers, which seems to require more workarounds
centos-nvhpc7: ubuntu-nvhpc7:
if: ${{ false }} # JOB DISABLED (NEEDS WORK): https://github.com/pybind/pybind11/issues/4690 runs-on: ubuntu-20.04
runs-on: ubuntu-latest name: "🐍 3 • NVHPC 23.5 • C++17 • x64"
name: "🐍 3 • CentOS7 / PGI 22.9 • x64"
container: centos:7
env:
# tzdata will try to ask for the timezone, so set the DEBIAN_FRONTEND
DEBIAN_FRONTEND: 'noninteractive'
steps: steps:
- uses: actions/checkout@v3 - uses: actions/checkout@v3
- name: Add Python 3 and a few requirements - name: Add NVHPC Repo
run: yum update -y && yum install -y epel-release && yum install -y git python3-devel make environment-modules cmake3 yum-utils run: |
echo 'deb [trusted=yes] https://developer.download.nvidia.com/hpc-sdk/ubuntu/amd64 /' | \
sudo tee /etc/apt/sources.list.d/nvhpc.list
- name: Install NVidia HPC SDK - name: Install 🐍 3 & NVHPC
run: yum-config-manager --add-repo https://developer.download.nvidia.com/hpc-sdk/rhel/nvhpc.repo && yum -y install nvhpc-22.9 run: |
sudo apt-get update -y && \
sudo apt-get install -y cmake environment-modules git python3-dev python3-pip python3-numpy && \
sudo apt-get install -y --no-install-recommends nvhpc-23-5 && \
sudo rm -rf /var/lib/apt/lists/*
python3 -m pip install --upgrade pip
python3 -m pip install --upgrade pytest
# On CentOS 7, we have to filter a few tests (compiler internal error) # On some systems, you many need further workarounds:
# and allow deeper template recursion (not needed on CentOS 8 with a newer
# standard library). On some systems, you many need further workarounds:
# https://github.com/pybind/pybind11/pull/2475 # https://github.com/pybind/pybind11/pull/2475
- name: Configure - name: Configure
shell: bash shell: bash
run: | run: |
source /etc/profile.d/modules.sh source /etc/profile.d/modules.sh
module load /opt/nvidia/hpc_sdk/modulefiles/nvhpc/22.9 module load /opt/nvidia/hpc_sdk/modulefiles/nvhpc/23.5
cmake3 -S . -B build -DDOWNLOAD_CATCH=ON \ cmake -S . -B build -DDOWNLOAD_CATCH=ON \
-DCMAKE_CXX_STANDARD=11 \ -DCMAKE_CXX_STANDARD=17 \
-DPYTHON_EXECUTABLE=$(python3 -c "import sys; print(sys.executable)") \ -DPYTHON_EXECUTABLE=$(python3 -c "import sys; print(sys.executable)") \
-DCMAKE_CXX_FLAGS="-Wc,--pending_instantiations=0" \ -DCMAKE_CXX_FLAGS="-Wc,--pending_instantiations=0" \
-DPYBIND11_TEST_FILTER="test_smart_ptr.cpp" -DPYBIND11_TEST_FILTER="test_smart_ptr.cpp"
# Building before installing Pip should produce a warning but not an error
- name: Build - name: Build
run: cmake3 --build build -j 2 --verbose run: cmake --build build -j 2 --verbose
- name: Install CMake with pip
run: |
python3 -m pip install --upgrade pip
python3 -m pip install pytest
- name: Python tests - name: Python tests
run: cmake3 --build build --target pytest run: cmake --build build --target pytest
- name: C++ tests - name: C++ tests
run: cmake3 --build build --target cpptest run: cmake --build build --target cpptest
- name: Interface test - name: Interface test
run: cmake3 --build build --target test_cmake_build run: cmake --build build --target test_cmake_build
# Testing on GCC using the GCC docker images (only recent images supported) # Testing on GCC using the GCC docker images (only recent images supported)

View File

@ -127,8 +127,7 @@ before a global translator is tried.
Inside the translator, ``std::rethrow_exception`` should be used within Inside the translator, ``std::rethrow_exception`` should be used within
a try block to re-throw the exception. One or more catch clauses to catch a try block to re-throw the exception. One or more catch clauses to catch
the appropriate exceptions should then be used with each clause using the appropriate exceptions should then be used with each clause using
``PyErr_SetString`` to set a Python exception or ``ex(string)`` to set ``py::set_error()`` (see below).
the python exception to a custom exception type (see below).
To declare a custom Python exception type, declare a ``py::exception`` variable To declare a custom Python exception type, declare a ``py::exception`` variable
and use this in the associated exception translator (note: it is often useful and use this in the associated exception translator (note: it is often useful
@ -142,14 +141,17 @@ standard python RuntimeError:
.. code-block:: cpp .. code-block:: cpp
static py::exception<MyCustomException> exc(m, "MyCustomError"); // This is a static object, so we must leak the Python reference:
// It is undefined when the destructor will run, possibly only after the
// Python interpreter is finalized already.
static py::handle exc = py::exception<MyCustomException>(m, "MyCustomError").release();
py::register_exception_translator([](std::exception_ptr p) { py::register_exception_translator([](std::exception_ptr p) {
try { try {
if (p) std::rethrow_exception(p); if (p) std::rethrow_exception(p);
} catch (const MyCustomException &e) { } catch (const MyCustomException &e) {
exc(e.what()); py::set_error(exc, e.what());
} catch (const OtherException &e) { } catch (const OtherException &e) {
PyErr_SetString(PyExc_RuntimeError, e.what()); py::set_error(PyExc_RuntimeError, e.what());
} }
}); });
@ -168,8 +170,7 @@ section.
.. note:: .. note::
Call either ``PyErr_SetString`` or a custom exception's call Call ``py::set_error()`` for every exception caught in a custom exception
operator (``exc(string)``) for every exception caught in a custom exception
translator. Failure to do so will cause Python to crash with ``SystemError: translator. Failure to do so will cause Python to crash with ``SystemError:
error return without exception set``. error return without exception set``.
@ -200,7 +201,7 @@ If module1 has the following translator:
try { try {
if (p) std::rethrow_exception(p); if (p) std::rethrow_exception(p);
} catch (const std::invalid_argument &e) { } catch (const std::invalid_argument &e) {
PyErr_SetString("module1 handled this") py::set_error(PyExc_ArgumentError, "module1 handled this");
} }
} }
@ -212,7 +213,7 @@ and module2 has the following similar translator:
try { try {
if (p) std::rethrow_exception(p); if (p) std::rethrow_exception(p);
} catch (const std::invalid_argument &e) { } catch (const std::invalid_argument &e) {
PyErr_SetString("module2 handled this") py::set_error(PyExc_ArgumentError, "module2 handled this");
} }
} }
@ -312,11 +313,11 @@ error protocol, which is outlined here.
After calling the Python C API, if Python returns an error, After calling the Python C API, if Python returns an error,
``throw py::error_already_set();``, which allows pybind11 to deal with the ``throw py::error_already_set();``, which allows pybind11 to deal with the
exception and pass it back to the Python interpreter. This includes calls to exception and pass it back to the Python interpreter. This includes calls to
the error setting functions such as ``PyErr_SetString``. the error setting functions such as ``py::set_error()``.
.. code-block:: cpp .. code-block:: cpp
PyErr_SetString(PyExc_TypeError, "C API type error demo"); py::set_error(PyExc_TypeError, "C API type error demo");
throw py::error_already_set(); throw py::error_already_set();
// But it would be easier to simply... // But it would be easier to simply...

View File

@ -375,7 +375,7 @@ extern "C" inline PyObject *pybind11_object_new(PyTypeObject *type, PyObject *,
extern "C" inline int pybind11_object_init(PyObject *self, PyObject *, PyObject *) { extern "C" inline int pybind11_object_init(PyObject *self, PyObject *, PyObject *) {
PyTypeObject *type = Py_TYPE(self); PyTypeObject *type = Py_TYPE(self);
std::string msg = get_fully_qualified_tp_name(type) + ": No constructor defined!"; std::string msg = get_fully_qualified_tp_name(type) + ": No constructor defined!";
PyErr_SetString(PyExc_TypeError, msg.c_str()); set_error(PyExc_TypeError, msg.c_str());
return -1; return -1;
} }
@ -581,7 +581,7 @@ extern "C" inline int pybind11_getbuffer(PyObject *obj, Py_buffer *view, int fla
if (view) { if (view) {
view->obj = nullptr; view->obj = nullptr;
} }
PyErr_SetString(PyExc_BufferError, "pybind11_getbuffer(): Internal error"); set_error(PyExc_BufferError, "pybind11_getbuffer(): Internal error");
return -1; return -1;
} }
std::memset(view, 0, sizeof(Py_buffer)); std::memset(view, 0, sizeof(Py_buffer));
@ -589,7 +589,7 @@ extern "C" inline int pybind11_getbuffer(PyObject *obj, Py_buffer *view, int fla
if ((flags & PyBUF_WRITABLE) == PyBUF_WRITABLE && info->readonly) { if ((flags & PyBUF_WRITABLE) == PyBUF_WRITABLE && info->readonly) {
delete info; delete info;
// view->obj = nullptr; // Was just memset to 0, so not necessary // view->obj = nullptr; // Was just memset to 0, so not necessary
PyErr_SetString(PyExc_BufferError, "Writable buffer requested for readonly storage"); set_error(PyExc_BufferError, "Writable buffer requested for readonly storage");
return -1; return -1;
} }
view->obj = obj; view->obj = obj;

View File

@ -399,7 +399,7 @@ PYBIND11_WARNING_POP
return nullptr; \ return nullptr; \
} \ } \
catch (const std::exception &e) { \ catch (const std::exception &e) { \
PyErr_SetString(PyExc_ImportError, e.what()); \ ::pybind11::set_error(PyExc_ImportError, e.what()); \
return nullptr; \ return nullptr; \
} }

View File

@ -363,7 +363,7 @@ inline bool raise_err(PyObject *exc_type, const char *msg) {
raise_from(exc_type, msg); raise_from(exc_type, msg);
return true; return true;
} }
PyErr_SetString(exc_type, msg); set_error(exc_type, msg);
return false; return false;
} }

View File

@ -790,7 +790,7 @@ public:
std::string tname = rtti_type ? rtti_type->name() : cast_type.name(); std::string tname = rtti_type ? rtti_type->name() : cast_type.name();
detail::clean_type_id(tname); detail::clean_type_id(tname);
std::string msg = "Unregistered type : " + tname; std::string msg = "Unregistered type : " + tname;
PyErr_SetString(PyExc_TypeError, msg.c_str()); set_error(PyExc_TypeError, msg.c_str());
return {nullptr, nullptr}; return {nullptr, nullptr};
} }

View File

@ -1008,7 +1008,7 @@ protected:
/// Create array from any object -- always returns a new reference /// Create array from any object -- always returns a new reference
static PyObject *raw_array(PyObject *ptr, int ExtraFlags = 0) { static PyObject *raw_array(PyObject *ptr, int ExtraFlags = 0) {
if (ptr == nullptr) { if (ptr == nullptr) {
PyErr_SetString(PyExc_ValueError, "cannot create a pybind11::array from a nullptr"); set_error(PyExc_ValueError, "cannot create a pybind11::array from a nullptr");
return nullptr; return nullptr;
} }
return detail::npy_api::get().PyArray_FromAny_( return detail::npy_api::get().PyArray_FromAny_(
@ -1155,7 +1155,7 @@ protected:
/// Create array from any object -- always returns a new reference /// Create array from any object -- always returns a new reference
static PyObject *raw_array_t(PyObject *ptr) { static PyObject *raw_array_t(PyObject *ptr) {
if (ptr == nullptr) { if (ptr == nullptr) {
PyErr_SetString(PyExc_ValueError, "cannot create a pybind11::array_t from a nullptr"); set_error(PyExc_ValueError, "cannot create a pybind11::array_t from a nullptr");
return nullptr; return nullptr;
} }
return detail::npy_api::get().PyArray_FromAny_(ptr, return detail::npy_api::get().PyArray_FromAny_(ptr,

View File

@ -695,9 +695,8 @@ protected:
if (overloads->is_constructor) { if (overloads->is_constructor) {
if (!parent if (!parent
|| !PyObject_TypeCheck(parent.ptr(), (PyTypeObject *) overloads->scope.ptr())) { || !PyObject_TypeCheck(parent.ptr(), (PyTypeObject *) overloads->scope.ptr())) {
PyErr_SetString( set_error(PyExc_TypeError,
PyExc_TypeError, "__init__(self, ...) called with invalid or missing `self` argument");
"__init__(self, ...) called with invalid or missing `self` argument");
return nullptr; return nullptr;
} }
@ -1008,7 +1007,7 @@ protected:
A translator may choose to do one of the following: A translator may choose to do one of the following:
- catch the exception and call PyErr_SetString or PyErr_SetObject - catch the exception and call py::set_error()
to set a standard (or custom) Python exception, or to set a standard (or custom) Python exception, or
- do nothing and let the exception fall through to the next translator, or - do nothing and let the exception fall through to the next translator, or
- delegate translation to the next translator by throwing a new type of exception. - delegate translation to the next translator by throwing a new type of exception.
@ -1024,8 +1023,7 @@ protected:
return nullptr; return nullptr;
} }
PyErr_SetString(PyExc_SystemError, set_error(PyExc_SystemError, "Exception escaped from default exception translator!");
"Exception escaped from default exception translator!");
return nullptr; return nullptr;
} }
@ -1126,7 +1124,7 @@ protected:
raise_from(PyExc_TypeError, msg.c_str()); raise_from(PyExc_TypeError, msg.c_str());
return nullptr; return nullptr;
} }
PyErr_SetString(PyExc_TypeError, msg.c_str()); set_error(PyExc_TypeError, msg.c_str());
return nullptr; return nullptr;
} }
if (!result) { if (!result) {
@ -1139,7 +1137,7 @@ protected:
raise_from(PyExc_TypeError, msg.c_str()); raise_from(PyExc_TypeError, msg.c_str());
return nullptr; return nullptr;
} }
PyErr_SetString(PyExc_TypeError, msg.c_str()); set_error(PyExc_TypeError, msg.c_str());
return nullptr; return nullptr;
} }
if (overloads->is_constructor && !self_value_and_holder.holder_constructed()) { if (overloads->is_constructor && !self_value_and_holder.holder_constructed()) {
@ -2788,7 +2786,7 @@ inline void register_local_exception_translator(ExceptionTranslator &&translator
/** /**
* Wrapper to generate a new Python exception type. * Wrapper to generate a new Python exception type.
* *
* This should only be used with PyErr_SetString for now. * This should only be used with py::set_error() for now.
* It is not (yet) possible to use as a py::base. * It is not (yet) possible to use as a py::base.
* Template type argument is reserved for future use. * Template type argument is reserved for future use.
*/ */
@ -2809,7 +2807,9 @@ public:
} }
// Sets the current python exception to this exception object with the given message // Sets the current python exception to this exception object with the given message
void operator()(const char *message) { PyErr_SetString(m_ptr, message); } PYBIND11_DEPRECATED("Please use py::set_error() instead "
"(https://github.com/pybind/pybind11/pull/4772)")
void operator()(const char *message) const { set_error(*this, message); }
}; };
PYBIND11_NAMESPACE_BEGIN(detail) PYBIND11_NAMESPACE_BEGIN(detail)
@ -2841,7 +2841,7 @@ register_exception_impl(handle scope, const char *name, handle base, bool isLoca
try { try {
std::rethrow_exception(p); std::rethrow_exception(p);
} catch (const CppException &e) { } catch (const CppException &e) {
detail::get_exception_object<CppException>()(e.what()); set_error(detail::get_exception_object<CppException>(), e.what());
} }
}); });
return ex; return ex;

View File

@ -334,6 +334,14 @@ public:
#endif #endif
}; };
inline void set_error(const handle &type, const char *message) {
PyErr_SetString(type.ptr(), message);
}
inline void set_error(const handle &type, const handle &value) {
PyErr_SetObject(type.ptr(), value.ptr());
}
/** \rst /** \rst
Holds a reference to a Python object (with reference counting) Holds a reference to a Python object (with reference counting)

View File

@ -16,12 +16,12 @@ namespace {
namespace py = pybind11; namespace py = pybind11;
void interleaved_error_already_set() { void interleaved_error_already_set() {
PyErr_SetString(PyExc_RuntimeError, "1st error."); py::set_error(PyExc_RuntimeError, "1st error.");
try { try {
throw py::error_already_set(); throw py::error_already_set();
} catch (const py::error_already_set &) { } catch (const py::error_already_set &) {
// The 2nd error could be conditional in a real application. // The 2nd error could be conditional in a real application.
PyErr_SetString(PyExc_RuntimeError, "2nd error."); py::set_error(PyExc_RuntimeError, "2nd error.");
} // Here the 1st error is destroyed before the 2nd error is fetched. } // Here the 1st error is destroyed before the 2nd error is fetched.
// The error_already_set dtor triggers a pybind11::detail::get_internals() // The error_already_set dtor triggers a pybind11::detail::get_internals()
// call via pybind11::gil_scoped_acquire. // call via pybind11::gil_scoped_acquire.

View File

@ -31,11 +31,11 @@ PYBIND11_MODULE(pybind11_cross_module_tests, m) {
// test_exceptions.py // test_exceptions.py
py::register_local_exception<LocalSimpleException>(m, "LocalSimpleException"); py::register_local_exception<LocalSimpleException>(m, "LocalSimpleException");
m.def("raise_runtime_error", []() { m.def("raise_runtime_error", []() {
PyErr_SetString(PyExc_RuntimeError, "My runtime error"); py::set_error(PyExc_RuntimeError, "My runtime error");
throw py::error_already_set(); throw py::error_already_set();
}); });
m.def("raise_value_error", []() { m.def("raise_value_error", []() {
PyErr_SetString(PyExc_ValueError, "My value error"); py::set_error(PyExc_ValueError, "My value error");
throw py::error_already_set(); throw py::error_already_set();
}); });
m.def("throw_pybind_value_error", []() { throw py::value_error("pybind11 value error"); }); m.def("throw_pybind_value_error", []() { throw py::value_error("pybind11 value error"); });
@ -49,7 +49,7 @@ PYBIND11_MODULE(pybind11_cross_module_tests, m) {
std::rethrow_exception(p); std::rethrow_exception(p);
} }
} catch (const shared_exception &e) { } catch (const shared_exception &e) {
PyErr_SetString(PyExc_KeyError, e.what()); py::set_error(PyExc_KeyError, e.what());
} }
}); });
@ -60,7 +60,7 @@ PYBIND11_MODULE(pybind11_cross_module_tests, m) {
std::rethrow_exception(p); std::rethrow_exception(p);
} }
} catch (const LocalException &e) { } catch (const LocalException &e) {
PyErr_SetString(PyExc_KeyError, e.what()); py::set_error(PyExc_KeyError, e.what());
} }
}); });

View File

@ -25,6 +25,10 @@ private:
std::string message = ""; std::string message = "";
}; };
class MyExceptionUseDeprecatedOperatorCall : public MyException {
using MyException::MyException;
};
// A type that should be translated to a standard Python exception // A type that should be translated to a standard Python exception
class MyException2 : public std::exception { class MyException2 : public std::exception {
public: public:
@ -109,8 +113,8 @@ TEST_SUBMODULE(exceptions, m) {
m.def("throw_std_exception", m.def("throw_std_exception",
[]() { throw std::runtime_error("This exception was intentionally thrown."); }); []() { throw std::runtime_error("This exception was intentionally thrown."); });
// make a new custom exception and use it as a translation target // PLEASE KEEP IN SYNC with docs/advanced/exceptions.rst
static py::exception<MyException> ex(m, "MyException"); static py::handle ex = py::exception<MyException>(m, "MyException").release();
py::register_exception_translator([](std::exception_ptr p) { py::register_exception_translator([](std::exception_ptr p) {
try { try {
if (p) { if (p) {
@ -118,7 +122,32 @@ TEST_SUBMODULE(exceptions, m) {
} }
} catch (const MyException &e) { } catch (const MyException &e) {
// Set MyException as the active python error // Set MyException as the active python error
ex(e.what()); py::set_error(ex, e.what());
}
});
// Same as above, but using the deprecated `py::exception<>::operator()`
// We want to be sure it still works, until it's removed.
static const auto *const exd = new py::exception<MyExceptionUseDeprecatedOperatorCall>(
m, "MyExceptionUseDeprecatedOperatorCall");
py::register_exception_translator([](std::exception_ptr p) {
try {
if (p) {
std::rethrow_exception(p);
}
} catch (const MyExceptionUseDeprecatedOperatorCall &e) {
#if defined(__INTEL_COMPILER) || defined(__NVCOMPILER)
// It is not worth the trouble dealing with warning suppressions for these compilers.
// Falling back to the recommended approach to keep the test code simple.
py::set_error(*exd, e.what());
#else
PYBIND11_WARNING_PUSH
PYBIND11_WARNING_DISABLE_CLANG("-Wdeprecated-declarations")
PYBIND11_WARNING_DISABLE_GCC("-Wdeprecated-declarations")
PYBIND11_WARNING_DISABLE_MSVC(4996)
(*exd)(e.what());
PYBIND11_WARNING_POP
#endif
} }
}); });
@ -132,7 +161,7 @@ TEST_SUBMODULE(exceptions, m) {
} }
} catch (const MyException2 &e) { } catch (const MyException2 &e) {
// Translate this exception to a standard RuntimeError // Translate this exception to a standard RuntimeError
PyErr_SetString(PyExc_RuntimeError, e.what()); py::set_error(PyExc_RuntimeError, e.what());
} }
}); });
@ -162,11 +191,16 @@ TEST_SUBMODULE(exceptions, m) {
std::rethrow_exception(p); std::rethrow_exception(p);
} }
} catch (const MyException6 &e) { } catch (const MyException6 &e) {
PyErr_SetString(PyExc_RuntimeError, e.what()); py::set_error(PyExc_RuntimeError, e.what());
} }
}); });
m.def("throws1", []() { throw MyException("this error should go to a custom type"); }); m.def("throws1",
[]() { throw MyException("this error should go to py::exception<MyException>"); });
m.def("throws1d", []() {
throw MyExceptionUseDeprecatedOperatorCall(
"this error should go to py::exception<MyExceptionUseDeprecatedOperatorCall>");
});
m.def("throws2", m.def("throws2",
[]() { throw MyException2("this error should go to a standard Python exception"); }); []() { throw MyException2("this error should go to a standard Python exception"); });
m.def("throws3", []() { throw MyException3("this error cannot be translated"); }); m.def("throws3", []() { throw MyException3("this error cannot be translated"); });
@ -222,7 +256,7 @@ TEST_SUBMODULE(exceptions, m) {
m.def("throw_already_set", [](bool err) { m.def("throw_already_set", [](bool err) {
if (err) { if (err) {
PyErr_SetString(PyExc_ValueError, "foo"); py::set_error(PyExc_ValueError, "foo");
} }
try { try {
throw py::error_already_set(); throw py::error_already_set();
@ -238,7 +272,7 @@ TEST_SUBMODULE(exceptions, m) {
} }
PyErr_Clear(); PyErr_Clear();
if (err) { if (err) {
PyErr_SetString(PyExc_ValueError, "foo"); py::set_error(PyExc_ValueError, "foo");
} }
throw py::error_already_set(); throw py::error_already_set();
}); });
@ -247,7 +281,7 @@ TEST_SUBMODULE(exceptions, m) {
bool retval = false; bool retval = false;
try { try {
PythonCallInDestructor set_dict_in_destructor(d); PythonCallInDestructor set_dict_in_destructor(d);
PyErr_SetString(PyExc_ValueError, "foo"); py::set_error(PyExc_ValueError, "foo");
throw py::error_already_set(); throw py::error_already_set();
} catch (const py::error_already_set &) { } catch (const py::error_already_set &) {
retval = true; retval = true;
@ -282,14 +316,14 @@ TEST_SUBMODULE(exceptions, m) {
m.def("throw_should_be_translated_to_key_error", []() { throw shared_exception(); }); m.def("throw_should_be_translated_to_key_error", []() { throw shared_exception(); });
m.def("raise_from", []() { m.def("raise_from", []() {
PyErr_SetString(PyExc_ValueError, "inner"); py::set_error(PyExc_ValueError, "inner");
py::raise_from(PyExc_ValueError, "outer"); py::raise_from(PyExc_ValueError, "outer");
throw py::error_already_set(); throw py::error_already_set();
}); });
m.def("raise_from_already_set", []() { m.def("raise_from_already_set", []() {
try { try {
PyErr_SetString(PyExc_ValueError, "inner"); py::set_error(PyExc_ValueError, "inner");
throw py::error_already_set(); throw py::error_already_set();
} catch (py::error_already_set &e) { } catch (py::error_already_set &e) {
py::raise_from(e, PyExc_ValueError, "outer"); py::raise_from(e, PyExc_ValueError, "outer");
@ -306,7 +340,7 @@ TEST_SUBMODULE(exceptions, m) {
}); });
m.def("error_already_set_what", [](const py::object &exc_type, const py::object &exc_value) { m.def("error_already_set_what", [](const py::object &exc_type, const py::object &exc_value) {
PyErr_SetObject(exc_type.ptr(), exc_value.ptr()); py::set_error(exc_type, exc_value);
std::string what = py::error_already_set().what(); std::string what = py::error_already_set().what();
bool py_err_set_after_what = (PyErr_Occurred() != nullptr); bool py_err_set_after_what = (PyErr_Occurred() != nullptr);
PyErr_Clear(); PyErr_Clear();
@ -321,7 +355,7 @@ TEST_SUBMODULE(exceptions, m) {
}); });
m.def("test_error_already_set_double_restore", [](bool dry_run) { m.def("test_error_already_set_double_restore", [](bool dry_run) {
PyErr_SetString(PyExc_ValueError, "Random error."); py::set_error(PyExc_ValueError, "Random error.");
py::error_already_set e; py::error_already_set e;
e.restore(); e.restore();
PyErr_Clear(); PyErr_Clear();

View File

@ -9,5 +9,5 @@ class PYBIND11_EXPORT_EXCEPTION shared_exception : public pybind11::builtin_exce
public: public:
using builtin_exception::builtin_exception; using builtin_exception::builtin_exception;
explicit shared_exception() : shared_exception("") {} explicit shared_exception() : shared_exception("") {}
void set_error() const override { PyErr_SetString(PyExc_RuntimeError, what()); } void set_error() const override { py::set_error(PyExc_RuntimeError, what()); }
}; };

View File

@ -139,7 +139,15 @@ def test_custom(msg):
# Can we catch a MyException? # Can we catch a MyException?
with pytest.raises(m.MyException) as excinfo: with pytest.raises(m.MyException) as excinfo:
m.throws1() m.throws1()
assert msg(excinfo.value) == "this error should go to a custom type" assert msg(excinfo.value) == "this error should go to py::exception<MyException>"
# Can we catch a MyExceptionUseDeprecatedOperatorCall?
with pytest.raises(m.MyExceptionUseDeprecatedOperatorCall) as excinfo:
m.throws1d()
assert (
msg(excinfo.value)
== "this error should go to py::exception<MyExceptionUseDeprecatedOperatorCall>"
)
# Can we translate to standard Python exceptions? # Can we translate to standard Python exceptions?
with pytest.raises(RuntimeError) as excinfo: with pytest.raises(RuntimeError) as excinfo:

View File

@ -25,7 +25,7 @@ PyObject *conv(PyObject *o) {
ret = PyFloat_FromDouble(v); ret = PyFloat_FromDouble(v);
} }
} else { } else {
PyErr_SetString(PyExc_TypeError, "Unexpected type"); py::set_error(PyExc_TypeError, "Unexpected type");
} }
return ret; return ret;
} }

View File

@ -78,14 +78,14 @@ TEST_SUBMODULE(type_caster_pyobject_ptr, m) {
m.def("cast_to_pyobject_ptr_nullptr", [](bool set_error) { m.def("cast_to_pyobject_ptr_nullptr", [](bool set_error) {
if (set_error) { if (set_error) {
PyErr_SetString(PyExc_RuntimeError, "Reflective of healthy error handling."); py::set_error(PyExc_RuntimeError, "Reflective of healthy error handling.");
} }
PyObject *ptr = nullptr; PyObject *ptr = nullptr;
py::cast(ptr); py::cast(ptr);
}); });
m.def("cast_to_pyobject_ptr_non_nullptr_with_error_set", []() { m.def("cast_to_pyobject_ptr_non_nullptr_with_error_set", []() {
PyErr_SetString(PyExc_RuntimeError, "Reflective of unhealthy error handling."); py::set_error(PyExc_RuntimeError, "Reflective of unhealthy error handling.");
py::cast(Py_None); py::cast(Py_None);
}); });

View File

@ -233,11 +233,13 @@ function(pybind11_add_module target_name)
endif() endif()
endif() endif()
# Use case-insensitive comparison to match the result of $<CONFIG:cfgs> if(DEFINED CMAKE_BUILD_TYPE) # see https://github.com/pybind/pybind11/issues/4454
string(TOUPPER "${CMAKE_BUILD_TYPE}" uppercase_CMAKE_BUILD_TYPE) # Use case-insensitive comparison to match the result of $<CONFIG:cfgs>
if(NOT MSVC AND NOT "${uppercase_CMAKE_BUILD_TYPE}" MATCHES DEBUG|RELWITHDEBINFO) string(TOUPPER "${CMAKE_BUILD_TYPE}" uppercase_CMAKE_BUILD_TYPE)
# Strip unnecessary sections of the binary on Linux/macOS if(NOT MSVC AND NOT "${uppercase_CMAKE_BUILD_TYPE}" MATCHES DEBUG|RELWITHDEBINFO)
pybind11_strip(${target_name}) # Strip unnecessary sections of the binary on Linux/macOS
pybind11_strip(${target_name})
endif()
endif() endif()
if(MSVC) if(MSVC)

View File

@ -212,10 +212,12 @@ function(pybind11_add_module target_name)
endif() endif()
endif() endif()
# Use case-insensitive comparison to match the result of $<CONFIG:cfgs> if(DEFINED CMAKE_BUILD_TYPE) # see https://github.com/pybind/pybind11/issues/4454
string(TOUPPER "${CMAKE_BUILD_TYPE}" uppercase_CMAKE_BUILD_TYPE) # Use case-insensitive comparison to match the result of $<CONFIG:cfgs>
if(NOT MSVC AND NOT "${uppercase_CMAKE_BUILD_TYPE}" MATCHES DEBUG|RELWITHDEBINFO) string(TOUPPER "${CMAKE_BUILD_TYPE}" uppercase_CMAKE_BUILD_TYPE)
pybind11_strip(${target_name}) if(NOT MSVC AND NOT "${uppercase_CMAKE_BUILD_TYPE}" MATCHES DEBUG|RELWITHDEBINFO)
pybind11_strip(${target_name})
endif()
endif() endif()
if(MSVC) if(MSVC)