Wenzel Jakob | 1c329aa | 2016-04-13 02:37:36 +0200 | [diff] [blame] | 1 | /* |
Dean Moldovan | a0c1ccf | 2016-08-12 13:50:00 +0200 | [diff] [blame] | 2 | tests/test_pickling.cpp -- pickle support |
Wenzel Jakob | 1c329aa | 2016-04-13 02:37:36 +0200 | [diff] [blame] | 3 | |
Wenzel Jakob | 8cb6cb3 | 2016-04-17 20:21:41 +0200 | [diff] [blame] | 4 | Copyright (c) 2016 Wenzel Jakob <wenzel.jakob@epfl.ch> |
Wenzel Jakob | 1c329aa | 2016-04-13 02:37:36 +0200 | [diff] [blame] | 5 | |
| 6 | All rights reserved. Use of this source code is governed by a |
| 7 | BSD-style license that can be found in the LICENSE file. |
| 8 | */ |
| 9 | |
Dean Moldovan | a0c1ccf | 2016-08-12 13:50:00 +0200 | [diff] [blame] | 10 | #include "pybind11_tests.h" |
Wenzel Jakob | 1c329aa | 2016-04-13 02:37:36 +0200 | [diff] [blame] | 11 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame^] | 12 | TEST_SUBMODULE(pickling, m) { |
| 13 | // test_roundtrip |
| 14 | class Pickleable { |
| 15 | public: |
| 16 | Pickleable(const std::string &value) : m_value(value) { } |
| 17 | const std::string &value() const { return m_value; } |
Wenzel Jakob | 1c329aa | 2016-04-13 02:37:36 +0200 | [diff] [blame] | 18 | |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame^] | 19 | void setExtra1(int extra1) { m_extra1 = extra1; } |
| 20 | void setExtra2(int extra2) { m_extra2 = extra2; } |
| 21 | int extra1() const { return m_extra1; } |
| 22 | int extra2() const { return m_extra2; } |
| 23 | private: |
| 24 | std::string m_value; |
| 25 | int m_extra1 = 0; |
| 26 | int m_extra2 = 0; |
| 27 | }; |
Wenzel Jakob | 1c329aa | 2016-04-13 02:37:36 +0200 | [diff] [blame] | 28 | py::class_<Pickleable>(m, "Pickleable") |
| 29 | .def(py::init<std::string>()) |
| 30 | .def("value", &Pickleable::value) |
| 31 | .def("extra1", &Pickleable::extra1) |
| 32 | .def("extra2", &Pickleable::extra2) |
| 33 | .def("setExtra1", &Pickleable::setExtra1) |
| 34 | .def("setExtra2", &Pickleable::setExtra2) |
| 35 | // For details on the methods below, refer to |
| 36 | // http://docs.python.org/3/library/pickle.html#pickling-class-instances |
| 37 | .def("__getstate__", [](const Pickleable &p) { |
| 38 | /* Return a tuple that fully encodes the state of the object */ |
| 39 | return py::make_tuple(p.value(), p.extra1(), p.extra2()); |
| 40 | }) |
| 41 | .def("__setstate__", [](Pickleable &p, py::tuple t) { |
| 42 | if (t.size() != 3) |
| 43 | throw std::runtime_error("Invalid state!"); |
| 44 | /* Invoke the constructor (need to use in-place version) */ |
| 45 | new (&p) Pickleable(t[0].cast<std::string>()); |
| 46 | |
| 47 | /* Assign any additional state */ |
| 48 | p.setExtra1(t[1].cast<int>()); |
| 49 | p.setExtra2(t[2].cast<int>()); |
| 50 | }); |
Dean Moldovan | 6fccf69 | 2016-10-11 01:12:48 +0200 | [diff] [blame] | 51 | |
Wenzel Jakob | 64cb699 | 2016-12-26 13:12:10 +0100 | [diff] [blame] | 52 | #if !defined(PYPY_VERSION) |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame^] | 53 | // test_roundtrip_with_dict |
| 54 | class PickleableWithDict { |
| 55 | public: |
| 56 | PickleableWithDict(const std::string &value) : value(value) { } |
| 57 | |
| 58 | std::string value; |
| 59 | int extra; |
| 60 | }; |
Dean Moldovan | 6fccf69 | 2016-10-11 01:12:48 +0200 | [diff] [blame] | 61 | py::class_<PickleableWithDict>(m, "PickleableWithDict", py::dynamic_attr()) |
| 62 | .def(py::init<std::string>()) |
| 63 | .def_readwrite("value", &PickleableWithDict::value) |
| 64 | .def_readwrite("extra", &PickleableWithDict::extra) |
| 65 | .def("__getstate__", [](py::object self) { |
| 66 | /* Also include __dict__ in state */ |
| 67 | return py::make_tuple(self.attr("value"), self.attr("extra"), self.attr("__dict__")); |
| 68 | }) |
| 69 | .def("__setstate__", [](py::object self, py::tuple t) { |
| 70 | if (t.size() != 3) |
| 71 | throw std::runtime_error("Invalid state!"); |
| 72 | /* Cast and construct */ |
| 73 | auto& p = self.cast<PickleableWithDict&>(); |
Wenzel Jakob | cd7eacc | 2017-01-04 09:01:59 -0500 | [diff] [blame] | 74 | new (&p) PickleableWithDict(t[0].cast<std::string>()); |
Dean Moldovan | 6fccf69 | 2016-10-11 01:12:48 +0200 | [diff] [blame] | 75 | |
| 76 | /* Assign C++ state */ |
| 77 | p.extra = t[1].cast<int>(); |
| 78 | |
| 79 | /* Assign Python state */ |
| 80 | self.attr("__dict__") = t[2]; |
| 81 | }); |
Wenzel Jakob | 64cb699 | 2016-12-26 13:12:10 +0100 | [diff] [blame] | 82 | #endif |
Jason Rhinelander | 391c754 | 2017-07-25 16:47:36 -0400 | [diff] [blame^] | 83 | } |