Skip to content

Commit f8a7212

Browse files
committed
added {dict,mapping}::contains method
1 parent c48b180 commit f8a7212

File tree

4 files changed

+33
-1
lines changed

4 files changed

+33
-1
lines changed

docs/api_core.rst

+11-1
Original file line numberDiff line numberDiff line change
@@ -670,7 +670,7 @@ Wrapper classes
670670

671671
Return the number of list elements.
672672

673-
.. cpp:function:: template <typename T> void append(T &&value)
673+
.. cpp:function:: template <typename T> void append(T&& value)
674674

675675
Append an element to the list. When `T` does not already represent a
676676
wrapped Python object, the function performs a cast.
@@ -707,6 +707,11 @@ Wrapper classes
707707

708708
Return the number of dictionary elements.
709709

710+
.. cpp:function:: template <typename T> bool contains(T&& key) const
711+
712+
Check whether the dictionary contains a particular key. When `T` does not
713+
already represent a wrapped Python object, the function performs a cast.
714+
710715
.. cpp:function:: detail::dict_iterator begin() const
711716

712717
Return an item iterator that returns ``std::pair<handle, handle>``
@@ -913,6 +918,11 @@ Wrapper classes
913918

914919
Wrapper class representing arbitrary Python mapping types.
915920

921+
.. cpp:function:: template <typename T> bool contains(T&& key) const
922+
923+
Check whether the map contains a particular key. When `T` does not
924+
already represent a wrapped Python object, the function performs a cast.
925+
916926
.. cpp:function:: list keys() const
917927

918928
Return a list containing all of the map's keys.

include/nanobind/nb_cast.h

+16
Original file line numberDiff line numberDiff line change
@@ -453,4 +453,20 @@ template <typename T> void list::append(T &&value) {
453453
detail::raise_python_error();
454454
}
455455

456+
template <typename T> bool dict::contains(T&& key) const {
457+
object o = nanobind::cast((detail::forward_t<T>) key);
458+
int rv = PyDict_Contains(m_ptr, o.ptr());
459+
if (rv == -1)
460+
detail::raise_python_error();
461+
return rv == 1;
462+
}
463+
464+
template <typename T> bool mapping::contains(T&& key) const {
465+
object o = nanobind::cast((detail::forward_t<T>) key);
466+
int rv = PyMapping_HasKey(m_ptr, o.ptr());
467+
if (rv == -1)
468+
detail::raise_python_error();
469+
return rv == 1;
470+
}
471+
456472
NAMESPACE_END(NB_NAMESPACE)

include/nanobind/nb_types.h

+2
Original file line numberDiff line numberDiff line change
@@ -463,6 +463,7 @@ class dict : public object {
463463
list keys() const { return steal<list>(detail::obj_op_1(m_ptr, PyDict_Keys)); }
464464
list values() const { return steal<list>(detail::obj_op_1(m_ptr, PyDict_Values)); }
465465
list items() const { return steal<list>(detail::obj_op_1(m_ptr, PyDict_Items)); }
466+
template <typename T> bool contains(T&& key) const;
466467
};
467468

468469
class sequence : public object {
@@ -474,6 +475,7 @@ class mapping : public object {
474475
list keys() const { return steal<list>(detail::obj_op_1(m_ptr, PyMapping_Keys)); }
475476
list values() const { return steal<list>(detail::obj_op_1(m_ptr, PyMapping_Values)); }
476477
list items() const { return steal<list>(detail::obj_op_1(m_ptr, PyMapping_Items)); }
478+
template <typename T> bool contains(T&& key) const;
477479
};
478480

479481
class args : public tuple {

tests/test_functions.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -138,6 +138,10 @@ NB_MODULE(test_functions_ext, m) {
138138
return result;
139139
});
140140

141+
m.def("test_10_contains", [](nb::dict d) {
142+
return d.contains(nb::str("foo"));
143+
});
144+
141145
// Test implicit conversion of various types
142146
m.def("test_11_sl", [](signed long x) { return x; });
143147
m.def("test_11_ul", [](unsigned long x) { return x; });

0 commit comments

Comments
 (0)