Skip to content

Commit

Permalink
Better error message when inheriting from a type without a constructor
Browse files Browse the repository at this point in the history
  • Loading branch information
virtuald committed Aug 24, 2020
1 parent 4493751 commit 773edfb
Show file tree
Hide file tree
Showing 3 changed files with 22 additions and 3 deletions.
10 changes: 7 additions & 3 deletions include/pybind11/detail/class.h
Original file line number Diff line number Diff line change
Expand Up @@ -171,8 +171,10 @@ extern "C" inline PyObject *pybind11_meta_call(PyObject *type, PyObject *args, P
// Ensure that the base __init__ function(s) were called
for (const auto &vh : values_and_holders(instance)) {
if (!vh.holder_constructed()) {
PyErr_Format(PyExc_TypeError, "%.200s.__init__() must be called when overriding __init__",
vh.type->type->tp_name);
auto message = (vh.type->type->tp_init == get_internals().default_object_init()) ?
"%.200s has no __init__ and cannot be used as a base class from Python" :
"%.200s.__init__() must be called when overriding __init__";
PyErr_Format(PyExc_TypeError, message, vh.type->type->tp_name);
Py_DECREF(self);
return nullptr;
}
Expand Down Expand Up @@ -302,6 +304,8 @@ extern "C" inline PyObject *pybind11_object_new(PyTypeObject *type, PyObject *,
/// An `__init__` function constructs the C++ object. Users should provide at least one
/// of these using `py::init` or directly with `.def(__init__, ...)`. Otherwise, the
/// following default function will be used which simply throws an exception.
///
/// Use `get_internals().default_object_init()` instead of using this function directly
extern "C" inline int pybind11_object_init(PyObject *self, PyObject *, PyObject *) {
PyTypeObject *type = Py_TYPE(self);
std::string msg;
Expand Down Expand Up @@ -620,7 +624,7 @@ inline PyObject* make_new_python_type(const type_record &rec) {
type->tp_bases = bases.release().ptr();

/* Don't inherit base __init__ */
type->tp_init = pybind11_object_init;
type->tp_init = internals.default_object_init();

/* Supported protocols */
type->tp_as_number = &heap_type->as_number;
Expand Down
4 changes: 4 additions & 0 deletions include/pybind11/detail/internals.h
Original file line number Diff line number Diff line change
Expand Up @@ -121,6 +121,10 @@ struct internals {
PYBIND11_TLS_FREE(tstate);
}
#endif

inline initproc default_object_init() {
return Py_TYPE(instance_base)->tp_init;
}
};

/// Additional type information which does not fit into the PyTypeObject.
Expand Down
11 changes: 11 additions & 0 deletions tests/test_class.py
Original file line number Diff line number Diff line change
Expand Up @@ -128,6 +128,17 @@ def __init__(self):
"Hamster.__init__() must be called when overriding __init__"] # PyPy
assert msg(exc_info.value) in expected

# Base doesn't have __init__
class ChChimera(m.Chimera):
def __init__(self):
pass

with pytest.raises(TypeError) as exc_info:
ChChimera()
expected = ["m.class_.Chimera has no __init__ and cannot be used as a base class from Python",
"Chimera has no __init__ and cannot be used as a base class from Python"] # PyPy
assert msg(exc_info.value) in expected


def test_automatic_upcasting():
assert type(m.return_class_1()).__name__ == "DerivedClass1"
Expand Down

0 comments on commit 773edfb

Please sign in to comment.