/* tests/test_pickling.cpp -- pickle support Copyright (c) 2016 Wenzel Jakob Copyright (c) 2021 The Pybind Development Team. All rights reserved. Use of this source code is governed by a BSD-style license that can be found in the LICENSE file. */ #include "pybind11_tests.h" #include #include #include namespace exercise_trampoline { struct SimpleBase { int num = 0; virtual ~SimpleBase() = default; // For compatibility with old clang versions: SimpleBase() = default; SimpleBase(const SimpleBase &) = default; }; struct SimpleBaseTrampoline : SimpleBase {}; struct SimpleCppDerived : SimpleBase {}; void wrap(py::module m) { py::class_(m, "SimpleBase") .def(py::init<>()) .def_readwrite("num", &SimpleBase::num) .def(py::pickle( [](const py::object &self) { py::dict d; if (py::hasattr(self, "__dict__")) { d = self.attr("__dict__"); } return py::make_tuple(self.attr("num"), d); }, [](const py::tuple &t) { if (t.size() != 2) { throw std::runtime_error("Invalid state!"); } auto cpp_state = std::unique_ptr(new SimpleBaseTrampoline); cpp_state->num = t[0].cast(); auto py_state = t[1].cast(); return std::make_pair(std::move(cpp_state), py_state); })); m.def("make_SimpleCppDerivedAsBase", []() { return std::unique_ptr(new SimpleCppDerived); }); m.def("check_dynamic_cast_SimpleCppDerived", [](const SimpleBase *base_ptr) { return dynamic_cast(base_ptr) != nullptr; }); } } // namespace exercise_trampoline TEST_SUBMODULE(pickling, m) { m.def("simple_callable", []() { return 20220426; }); // test_roundtrip class Pickleable { public: explicit Pickleable(const std::string &value) : m_value(value) {} const std::string &value() const { return m_value; } void setExtra1(int extra1) { m_extra1 = extra1; } void setExtra2(int extra2) { m_extra2 = extra2; } int extra1() const { return m_extra1; } int extra2() const { return m_extra2; } private: std::string m_value; int m_extra1 = 0; int m_extra2 = 0; }; class PickleableNew : public Pickleable { public: using Pickleable::Pickleable; }; py::class_ pyPickleable(m, "Pickleable"); pyPickleable.def(py::init()) .def("value", &Pickleable::value) .def("extra1", &Pickleable::extra1) .def("extra2", &Pickleable::extra2) .def("setExtra1", &Pickleable::setExtra1) .def("setExtra2", &Pickleable::setExtra2) // For details on the methods below, refer to // http://docs.python.org/3/library/pickle.html#pickling-class-instances .def("__getstate__", [](const Pickleable &p) { /* Return a tuple that fully encodes the state of the object */ return py::make_tuple(p.value(), p.extra1(), p.extra2()); }); ignoreOldStyleInitWarnings([&pyPickleable]() { pyPickleable.def("__setstate__", [](Pickleable &p, const py::tuple &t) { if (t.size() != 3) { throw std::runtime_error("Invalid state!"); } /* Invoke the constructor (need to use in-place version) */ new (&p) Pickleable(t[0].cast()); /* Assign any additional state */ p.setExtra1(t[1].cast()); p.setExtra2(t[2].cast()); }); }); py::class_(m, "PickleableNew") .def(py::init()) .def(py::pickle( [](const PickleableNew &p) { return py::make_tuple(p.value(), p.extra1(), p.extra2()); }, [](const py::tuple &t) { if (t.size() != 3) { throw std::runtime_error("Invalid state!"); } auto p = PickleableNew(t[0].cast()); p.setExtra1(t[1].cast()); p.setExtra2(t[2].cast()); return p; })); #if !defined(PYPY_VERSION) // test_roundtrip_with_dict class PickleableWithDict { public: explicit PickleableWithDict(const std::string &value) : value(value) {} std::string value; int extra; }; class PickleableWithDictNew : public PickleableWithDict { public: using PickleableWithDict::PickleableWithDict; }; py::class_ pyPickleableWithDict( m, "PickleableWithDict", py::dynamic_attr()); pyPickleableWithDict.def(py::init()) .def_readwrite("value", &PickleableWithDict::value) .def_readwrite("extra", &PickleableWithDict::extra) .def("__getstate__", [](const py::object &self) { /* Also include __dict__ in state */ return py::make_tuple(self.attr("value"), self.attr("extra"), self.attr("__dict__")); }); ignoreOldStyleInitWarnings([&pyPickleableWithDict]() { pyPickleableWithDict.def("__setstate__", [](const py::object &self, const py::tuple &t) { if (t.size() != 3) { throw std::runtime_error("Invalid state!"); } /* Cast and construct */ auto &p = self.cast(); new (&p) PickleableWithDict(t[0].cast()); /* Assign C++ state */ p.extra = t[1].cast(); /* Assign Python state */ self.attr("__dict__") = t[2]; }); }); py::class_(m, "PickleableWithDictNew") .def(py::init()) .def(py::pickle( [](const py::object &self) { return py::make_tuple( self.attr("value"), self.attr("extra"), self.attr("__dict__")); }, [](const py::tuple &t) { if (t.size() != 3) { throw std::runtime_error("Invalid state!"); } auto cpp_state = PickleableWithDictNew(t[0].cast()); cpp_state.extra = t[1].cast(); auto py_state = t[2].cast(); return std::make_pair(cpp_state, py_state); })); #endif exercise_trampoline::wrap(m); }