mirror of
https://github.com/pybind/pybind11.git
synced 2024-11-22 05:05:11 +00:00
Add make_simple_namespace function and tests (#2840)
Co-authored-by: Jouke Witteveen <j.witteveen@cosine.nl>
This commit is contained in:
parent
c8ce4b8df8
commit
031a700dfd
@ -20,6 +20,47 @@ Available types include :class:`handle`, :class:`object`, :class:`bool_`,
|
|||||||
Be sure to review the :ref:`pytypes_gotchas` before using this heavily in
|
Be sure to review the :ref:`pytypes_gotchas` before using this heavily in
|
||||||
your C++ API.
|
your C++ API.
|
||||||
|
|
||||||
|
.. _instantiating_compound_types:
|
||||||
|
|
||||||
|
Instantiating compound Python types from C++
|
||||||
|
============================================
|
||||||
|
|
||||||
|
Dictionaries can be initialized in the :class:`dict` constructor:
|
||||||
|
|
||||||
|
.. code-block:: cpp
|
||||||
|
|
||||||
|
using namespace pybind11::literals; // to bring in the `_a` literal
|
||||||
|
py::dict d("spam"_a=py::none(), "eggs"_a=42);
|
||||||
|
|
||||||
|
A tuple of python objects can be instantiated using :func:`py::make_tuple`:
|
||||||
|
|
||||||
|
.. code-block:: cpp
|
||||||
|
|
||||||
|
py::tuple tup = py::make_tuple(42, py::none(), "spam");
|
||||||
|
|
||||||
|
Each element is converted to a supported Python type.
|
||||||
|
|
||||||
|
A `simple namespace`_ can be instantiated using
|
||||||
|
:func:`py::make_simple_namespace`:
|
||||||
|
|
||||||
|
.. code-block:: cpp
|
||||||
|
|
||||||
|
using namespace pybind11::literals; // to bring in the `_a` literal
|
||||||
|
py::object ns = py::make_simple_namespace("spam"_a=py::none(), "eggs"_a=42);
|
||||||
|
|
||||||
|
Attributes on a namespace can be modified with the :func:`py::delattr`,
|
||||||
|
:func:`py::getattr`, and :func:`py::setattr` functions. Simple namespaces can
|
||||||
|
be useful as lightweight stand-ins for class instances.
|
||||||
|
|
||||||
|
.. note::
|
||||||
|
|
||||||
|
``make_simple_namespace`` is not available in Python 2.
|
||||||
|
|
||||||
|
.. versionchanged:: 2.8
|
||||||
|
``make_simple_namespace`` added.
|
||||||
|
|
||||||
|
.. _simple namespace: https://docs.python.org/3/library/types.html#types.SimpleNamespace
|
||||||
|
|
||||||
.. _casting_back_and_forth:
|
.. _casting_back_and_forth:
|
||||||
|
|
||||||
Casting back and forth
|
Casting back and forth
|
||||||
@ -30,7 +71,7 @@ types to Python, which can be done using :func:`py::cast`:
|
|||||||
|
|
||||||
.. code-block:: cpp
|
.. code-block:: cpp
|
||||||
|
|
||||||
MyClass *cls = ..;
|
MyClass *cls = ...;
|
||||||
py::object obj = py::cast(cls);
|
py::object obj = py::cast(cls);
|
||||||
|
|
||||||
The reverse direction uses the following syntax:
|
The reverse direction uses the following syntax:
|
||||||
|
@ -1018,6 +1018,16 @@ template <return_value_policy policy = return_value_policy::automatic_reference,
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if PY_VERSION_HEX >= 0x03030000
|
||||||
|
template <typename... Args,
|
||||||
|
typename = detail::enable_if_t<args_are_all_keyword_or_ds<Args...>()>>
|
||||||
|
object make_simple_namespace(Args&&... args_) {
|
||||||
|
PyObject *ns = _PyNamespace_New(dict(std::forward<Args>(args_)...).ptr());
|
||||||
|
if (!ns) throw error_already_set();
|
||||||
|
return reinterpret_steal<object>(ns);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
/// \ingroup annotations
|
/// \ingroup annotations
|
||||||
/// Annotation for arguments
|
/// Annotation for arguments
|
||||||
struct arg {
|
struct arg {
|
||||||
|
@ -70,6 +70,19 @@ TEST_SUBMODULE(pytypes, m) {
|
|||||||
m.def("dict_contains",
|
m.def("dict_contains",
|
||||||
[](const py::dict &dict, const char *val) { return dict.contains(val); });
|
[](const py::dict &dict, const char *val) { return dict.contains(val); });
|
||||||
|
|
||||||
|
// test_tuple
|
||||||
|
m.def("get_tuple", []() { return py::make_tuple(42, py::none(), "spam"); });
|
||||||
|
|
||||||
|
#if PY_VERSION_HEX >= 0x03030000
|
||||||
|
// test_simple_namespace
|
||||||
|
m.def("get_simple_namespace", []() {
|
||||||
|
auto ns = py::make_simple_namespace("attr"_a=42, "x"_a="foo", "wrong"_a=1);
|
||||||
|
py::delattr(ns, "wrong");
|
||||||
|
py::setattr(ns, "right", py::int_(2));
|
||||||
|
return ns;
|
||||||
|
});
|
||||||
|
#endif
|
||||||
|
|
||||||
// test_str
|
// test_str
|
||||||
m.def("str_from_string", []() { return py::str(std::string("baz")); });
|
m.def("str_from_string", []() { return py::str(std::string("baz")); });
|
||||||
m.def("str_from_bytes", []() { return py::str(py::bytes("boo", 3)); });
|
m.def("str_from_bytes", []() { return py::str(py::bytes("boo", 3)); });
|
||||||
|
@ -99,6 +99,19 @@ def test_dict(capture, doc):
|
|||||||
assert m.dict_keyword_constructor() == {"x": 1, "y": 2, "z": 3}
|
assert m.dict_keyword_constructor() == {"x": 1, "y": 2, "z": 3}
|
||||||
|
|
||||||
|
|
||||||
|
def test_tuple():
|
||||||
|
assert m.get_tuple() == (42, None, "spam")
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.skipif("env.PY2")
|
||||||
|
def test_simple_namespace():
|
||||||
|
ns = m.get_simple_namespace()
|
||||||
|
assert ns.attr == 42
|
||||||
|
assert ns.x == "foo"
|
||||||
|
assert ns.right == 2
|
||||||
|
assert not hasattr(ns, "wrong")
|
||||||
|
|
||||||
|
|
||||||
def test_str(doc):
|
def test_str(doc):
|
||||||
assert m.str_from_string().encode().decode() == "baz"
|
assert m.str_from_string().encode().decode() == "baz"
|
||||||
assert m.str_from_bytes().encode().decode() == "boo"
|
assert m.str_from_bytes().encode().decode() == "boo"
|
||||||
|
Loading…
Reference in New Issue
Block a user