1use crate::class::basic::CompareOp;
2use crate::conversion::{AsPyPointer, FromPyObjectBound, IntoPyObject};
3use crate::err::{DowncastError, DowncastIntoError, PyErr, PyResult};
4use crate::exceptions::{PyAttributeError, PyTypeError};
5use crate::ffi_ptr_ext::FfiPtrExt;
6use crate::instance::Bound;
7use crate::internal::get_slot::TP_DESCR_GET;
8use crate::internal_tricks::ptr_from_ref;
9use crate::py_result_ext::PyResultExt;
10use crate::type_object::{PyTypeCheck, PyTypeInfo};
11#[cfg(not(any(PyPy, GraalPy)))]
12use crate::types::PySuper;
13use crate::types::{PyDict, PyIterator, PyList, PyString, PyTuple, PyType};
14use crate::{err, ffi, Borrowed, BoundObject, IntoPyObjectExt, Python};
15use std::cell::UnsafeCell;
16use std::cmp::Ordering;
17use std::os::raw::c_int;
18
19#[doc = concat!("[the guide](https://pyo3.rs/v", env!("CARGO_PKG_VERSION"), "/types.html#concrete-python-types)")]
29#[repr(transparent)]
31pub struct PyAny(UnsafeCell<ffi::PyObject>);
32
33#[allow(non_snake_case)]
34fn PyObject_Check(_: *mut ffi::PyObject) -> c_int {
37 1
38}
39
40pyobject_native_type_info!(
41 PyAny,
42 pyobject_native_static_type_object!(ffi::PyBaseObject_Type),
43 Some("builtins"),
44 #checkfunction=PyObject_Check
45);
46
47pyobject_native_type_sized!(PyAny, ffi::PyObject);
48impl crate::impl_::pyclass::PyClassBaseType for PyAny {
50 type LayoutAsBase = crate::impl_::pycell::PyClassObjectBase<ffi::PyObject>;
51 type BaseNativeType = PyAny;
52 type Initializer = crate::impl_::pyclass_init::PyNativeTypeInitializer<Self>;
53 type PyClassMutability = crate::pycell::impl_::ImmutableClass;
54}
55
56#[doc(alias = "PyAny")]
61pub trait PyAnyMethods<'py>: crate::sealed::Sealed {
62 fn is<T: AsPyPointer>(&self, other: &T) -> bool;
67
68 fn hasattr<N>(&self, attr_name: N) -> PyResult<bool>
91 where
92 N: IntoPyObject<'py, Target = PyString>;
93
94 fn getattr<N>(&self, attr_name: N) -> PyResult<Bound<'py, PyAny>>
117 where
118 N: IntoPyObject<'py, Target = PyString>;
119
120 fn setattr<N, V>(&self, attr_name: N, value: V) -> PyResult<()>
143 where
144 N: IntoPyObject<'py, Target = PyString>,
145 V: IntoPyObject<'py>;
146
147 fn delattr<N>(&self, attr_name: N) -> PyResult<()>
154 where
155 N: IntoPyObject<'py, Target = PyString>;
156
157 fn compare<O>(&self, other: O) -> PyResult<Ordering>
204 where
205 O: IntoPyObject<'py>;
206
207 fn rich_compare<O>(&self, other: O, compare_op: CompareOp) -> PyResult<Bound<'py, PyAny>>
241 where
242 O: IntoPyObject<'py>;
243
244 fn neg(&self) -> PyResult<Bound<'py, PyAny>>;
248
249 fn pos(&self) -> PyResult<Bound<'py, PyAny>>;
253
254 fn abs(&self) -> PyResult<Bound<'py, PyAny>>;
258
259 fn bitnot(&self) -> PyResult<Bound<'py, PyAny>>;
261
262 fn lt<O>(&self, other: O) -> PyResult<bool>
266 where
267 O: IntoPyObject<'py>;
268
269 fn le<O>(&self, other: O) -> PyResult<bool>
273 where
274 O: IntoPyObject<'py>;
275
276 fn eq<O>(&self, other: O) -> PyResult<bool>
280 where
281 O: IntoPyObject<'py>;
282
283 fn ne<O>(&self, other: O) -> PyResult<bool>
287 where
288 O: IntoPyObject<'py>;
289
290 fn gt<O>(&self, other: O) -> PyResult<bool>
294 where
295 O: IntoPyObject<'py>;
296
297 fn ge<O>(&self, other: O) -> PyResult<bool>
301 where
302 O: IntoPyObject<'py>;
303
304 fn add<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
306 where
307 O: IntoPyObject<'py>;
308
309 fn sub<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
311 where
312 O: IntoPyObject<'py>;
313
314 fn mul<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
316 where
317 O: IntoPyObject<'py>;
318
319 fn matmul<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
321 where
322 O: IntoPyObject<'py>;
323
324 fn div<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
326 where
327 O: IntoPyObject<'py>;
328
329 fn floor_div<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
331 where
332 O: IntoPyObject<'py>;
333
334 fn rem<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
336 where
337 O: IntoPyObject<'py>;
338
339 fn divmod<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
341 where
342 O: IntoPyObject<'py>;
343
344 fn lshift<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
346 where
347 O: IntoPyObject<'py>;
348
349 fn rshift<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
351 where
352 O: IntoPyObject<'py>;
353
354 fn pow<O1, O2>(&self, other: O1, modulus: O2) -> PyResult<Bound<'py, PyAny>>
357 where
358 O1: IntoPyObject<'py>,
359 O2: IntoPyObject<'py>;
360
361 fn bitand<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
363 where
364 O: IntoPyObject<'py>;
365
366 fn bitor<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
368 where
369 O: IntoPyObject<'py>;
370
371 fn bitxor<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
373 where
374 O: IntoPyObject<'py>;
375
376 fn is_callable(&self) -> bool;
404
405 fn call<A>(&self, args: A, kwargs: Option<&Bound<'py, PyDict>>) -> PyResult<Bound<'py, PyAny>>
438 where
439 A: IntoPyObject<'py, Target = PyTuple>;
440
441 fn call0(&self) -> PyResult<Bound<'py, PyAny>>;
462
463 fn call1<A>(&self, args: A) -> PyResult<Bound<'py, PyAny>>
493 where
494 A: IntoPyObject<'py, Target = PyTuple>;
495
496 fn call_method<N, A>(
534 &self,
535 name: N,
536 args: A,
537 kwargs: Option<&Bound<'py, PyDict>>,
538 ) -> PyResult<Bound<'py, PyAny>>
539 where
540 N: IntoPyObject<'py, Target = PyString>,
541 A: IntoPyObject<'py, Target = PyTuple>;
542
543 fn call_method0<N>(&self, name: N) -> PyResult<Bound<'py, PyAny>>
577 where
578 N: IntoPyObject<'py, Target = PyString>;
579
580 fn call_method1<N, A>(&self, name: N, args: A) -> PyResult<Bound<'py, PyAny>>
615 where
616 N: IntoPyObject<'py, Target = PyString>,
617 A: IntoPyObject<'py, Target = PyTuple>;
618
619 fn is_truthy(&self) -> PyResult<bool>;
623
624 fn is_none(&self) -> bool;
628
629 #[deprecated(since = "0.23.0", note = "use `.is(py.Ellipsis())` instead")]
633 fn is_ellipsis(&self) -> bool;
634
635 fn is_empty(&self) -> PyResult<bool>;
639
640 fn get_item<K>(&self, key: K) -> PyResult<Bound<'py, PyAny>>
644 where
645 K: IntoPyObject<'py>;
646
647 fn set_item<K, V>(&self, key: K, value: V) -> PyResult<()>
651 where
652 K: IntoPyObject<'py>,
653 V: IntoPyObject<'py>;
654
655 fn del_item<K>(&self, key: K) -> PyResult<()>
659 where
660 K: IntoPyObject<'py>;
661
662 fn try_iter(&self) -> PyResult<Bound<'py, PyIterator>>;
687
688 #[deprecated(since = "0.23.0", note = "use `try_iter` instead")]
693 fn iter(&self) -> PyResult<Bound<'py, PyIterator>>;
694
695 fn get_type(&self) -> Bound<'py, PyType>;
697
698 fn get_type_ptr(&self) -> *mut ffi::PyTypeObject;
700
701 fn downcast<T>(&self) -> Result<&Bound<'py, T>, DowncastError<'_, 'py>>
754 where
755 T: PyTypeCheck;
756
757 fn downcast_into<T>(self) -> Result<Bound<'py, T>, DowncastIntoError<'py>>
780 where
781 T: PyTypeCheck;
782
783 fn downcast_exact<T>(&self) -> Result<&Bound<'py, T>, DowncastError<'_, 'py>>
814 where
815 T: PyTypeInfo;
816
817 fn downcast_into_exact<T>(self) -> Result<Bound<'py, T>, DowncastIntoError<'py>>
819 where
820 T: PyTypeInfo;
821
822 unsafe fn downcast_unchecked<T>(&self) -> &Bound<'py, T>;
828
829 unsafe fn downcast_into_unchecked<T>(self) -> Bound<'py, T>;
835
836 fn extract<'a, T>(&'a self) -> PyResult<T>
841 where
842 T: FromPyObjectBound<'a, 'py>;
843
844 fn get_refcnt(&self) -> isize;
846
847 fn repr(&self) -> PyResult<Bound<'py, PyString>>;
851
852 fn str(&self) -> PyResult<Bound<'py, PyString>>;
856
857 fn hash(&self) -> PyResult<isize>;
861
862 fn len(&self) -> PyResult<usize>;
866
867 fn dir(&self) -> PyResult<Bound<'py, PyList>>;
871
872 fn is_instance(&self, ty: &Bound<'py, PyAny>) -> PyResult<bool>;
876
877 fn is_exact_instance(&self, ty: &Bound<'py, PyAny>) -> bool;
881
882 fn is_instance_of<T: PyTypeInfo>(&self) -> bool;
887
888 fn is_exact_instance_of<T: PyTypeInfo>(&self) -> bool;
893
894 fn contains<V>(&self, value: V) -> PyResult<bool>
898 where
899 V: IntoPyObject<'py>;
900
901 #[cfg(not(any(PyPy, GraalPy)))]
905 fn py_super(&self) -> PyResult<Bound<'py, PySuper>>;
906}
907
908macro_rules! implement_binop {
909 ($name:ident, $c_api:ident, $op:expr) => {
910 #[doc = concat!("Computes `self ", $op, " other`.")]
911 fn $name<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
912 where
913 O: IntoPyObject<'py>,
914 {
915 fn inner<'py>(
916 any: &Bound<'py, PyAny>,
917 other: Borrowed<'_, 'py, PyAny>,
918 ) -> PyResult<Bound<'py, PyAny>> {
919 unsafe { ffi::$c_api(any.as_ptr(), other.as_ptr()).assume_owned_or_err(any.py()) }
920 }
921
922 let py = self.py();
923 inner(
924 self,
925 other.into_pyobject_or_pyerr(py)?.into_any().as_borrowed(),
926 )
927 }
928 };
929}
930
931impl<'py> PyAnyMethods<'py> for Bound<'py, PyAny> {
932 #[inline]
933 fn is<T: AsPyPointer>(&self, other: &T) -> bool {
934 self.as_ptr() == other.as_ptr()
935 }
936
937 fn hasattr<N>(&self, attr_name: N) -> PyResult<bool>
938 where
939 N: IntoPyObject<'py, Target = PyString>,
940 {
941 fn inner(py: Python<'_>, getattr_result: PyResult<Bound<'_, PyAny>>) -> PyResult<bool> {
944 match getattr_result {
945 Ok(_) => Ok(true),
946 Err(err) if err.is_instance_of::<PyAttributeError>(py) => Ok(false),
947 Err(e) => Err(e),
948 }
949 }
950
951 inner(self.py(), self.getattr(attr_name))
952 }
953
954 fn getattr<N>(&self, attr_name: N) -> PyResult<Bound<'py, PyAny>>
955 where
956 N: IntoPyObject<'py, Target = PyString>,
957 {
958 fn inner<'py>(
959 any: &Bound<'py, PyAny>,
960 attr_name: Borrowed<'_, '_, PyString>,
961 ) -> PyResult<Bound<'py, PyAny>> {
962 unsafe {
963 ffi::PyObject_GetAttr(any.as_ptr(), attr_name.as_ptr())
964 .assume_owned_or_err(any.py())
965 }
966 }
967
968 inner(
969 self,
970 attr_name
971 .into_pyobject(self.py())
972 .map_err(Into::into)?
973 .as_borrowed(),
974 )
975 }
976
977 fn setattr<N, V>(&self, attr_name: N, value: V) -> PyResult<()>
978 where
979 N: IntoPyObject<'py, Target = PyString>,
980 V: IntoPyObject<'py>,
981 {
982 fn inner(
983 any: &Bound<'_, PyAny>,
984 attr_name: Borrowed<'_, '_, PyString>,
985 value: Borrowed<'_, '_, PyAny>,
986 ) -> PyResult<()> {
987 err::error_on_minusone(any.py(), unsafe {
988 ffi::PyObject_SetAttr(any.as_ptr(), attr_name.as_ptr(), value.as_ptr())
989 })
990 }
991
992 let py = self.py();
993 inner(
994 self,
995 attr_name.into_pyobject_or_pyerr(py)?.as_borrowed(),
996 value.into_pyobject_or_pyerr(py)?.into_any().as_borrowed(),
997 )
998 }
999
1000 fn delattr<N>(&self, attr_name: N) -> PyResult<()>
1001 where
1002 N: IntoPyObject<'py, Target = PyString>,
1003 {
1004 fn inner(any: &Bound<'_, PyAny>, attr_name: Borrowed<'_, '_, PyString>) -> PyResult<()> {
1005 err::error_on_minusone(any.py(), unsafe {
1006 ffi::PyObject_DelAttr(any.as_ptr(), attr_name.as_ptr())
1007 })
1008 }
1009
1010 let py = self.py();
1011 inner(self, attr_name.into_pyobject_or_pyerr(py)?.as_borrowed())
1012 }
1013
1014 fn compare<O>(&self, other: O) -> PyResult<Ordering>
1015 where
1016 O: IntoPyObject<'py>,
1017 {
1018 fn inner(any: &Bound<'_, PyAny>, other: Borrowed<'_, '_, PyAny>) -> PyResult<Ordering> {
1019 let other = other.as_ptr();
1020 let do_compare = |other, op| unsafe {
1023 ffi::PyObject_RichCompare(any.as_ptr(), other, op)
1024 .assume_owned_or_err(any.py())
1025 .and_then(|obj| obj.is_truthy())
1026 };
1027 if do_compare(other, ffi::Py_EQ)? {
1028 Ok(Ordering::Equal)
1029 } else if do_compare(other, ffi::Py_LT)? {
1030 Ok(Ordering::Less)
1031 } else if do_compare(other, ffi::Py_GT)? {
1032 Ok(Ordering::Greater)
1033 } else {
1034 Err(PyTypeError::new_err(
1035 "PyAny::compare(): All comparisons returned false",
1036 ))
1037 }
1038 }
1039
1040 let py = self.py();
1041 inner(
1042 self,
1043 other.into_pyobject_or_pyerr(py)?.into_any().as_borrowed(),
1044 )
1045 }
1046
1047 fn rich_compare<O>(&self, other: O, compare_op: CompareOp) -> PyResult<Bound<'py, PyAny>>
1048 where
1049 O: IntoPyObject<'py>,
1050 {
1051 fn inner<'py>(
1052 any: &Bound<'py, PyAny>,
1053 other: Borrowed<'_, 'py, PyAny>,
1054 compare_op: CompareOp,
1055 ) -> PyResult<Bound<'py, PyAny>> {
1056 unsafe {
1057 ffi::PyObject_RichCompare(any.as_ptr(), other.as_ptr(), compare_op as c_int)
1058 .assume_owned_or_err(any.py())
1059 }
1060 }
1061
1062 let py = self.py();
1063 inner(
1064 self,
1065 other.into_pyobject_or_pyerr(py)?.into_any().as_borrowed(),
1066 compare_op,
1067 )
1068 }
1069
1070 fn neg(&self) -> PyResult<Bound<'py, PyAny>> {
1071 unsafe { ffi::PyNumber_Negative(self.as_ptr()).assume_owned_or_err(self.py()) }
1072 }
1073
1074 fn pos(&self) -> PyResult<Bound<'py, PyAny>> {
1075 fn inner<'py>(any: &Bound<'py, PyAny>) -> PyResult<Bound<'py, PyAny>> {
1076 unsafe { ffi::PyNumber_Positive(any.as_ptr()).assume_owned_or_err(any.py()) }
1077 }
1078
1079 inner(self)
1080 }
1081
1082 fn abs(&self) -> PyResult<Bound<'py, PyAny>> {
1083 fn inner<'py>(any: &Bound<'py, PyAny>) -> PyResult<Bound<'py, PyAny>> {
1084 unsafe { ffi::PyNumber_Absolute(any.as_ptr()).assume_owned_or_err(any.py()) }
1085 }
1086
1087 inner(self)
1088 }
1089
1090 fn bitnot(&self) -> PyResult<Bound<'py, PyAny>> {
1091 fn inner<'py>(any: &Bound<'py, PyAny>) -> PyResult<Bound<'py, PyAny>> {
1092 unsafe { ffi::PyNumber_Invert(any.as_ptr()).assume_owned_or_err(any.py()) }
1093 }
1094
1095 inner(self)
1096 }
1097
1098 fn lt<O>(&self, other: O) -> PyResult<bool>
1099 where
1100 O: IntoPyObject<'py>,
1101 {
1102 self.rich_compare(other, CompareOp::Lt)
1103 .and_then(|any| any.is_truthy())
1104 }
1105
1106 fn le<O>(&self, other: O) -> PyResult<bool>
1107 where
1108 O: IntoPyObject<'py>,
1109 {
1110 self.rich_compare(other, CompareOp::Le)
1111 .and_then(|any| any.is_truthy())
1112 }
1113
1114 fn eq<O>(&self, other: O) -> PyResult<bool>
1115 where
1116 O: IntoPyObject<'py>,
1117 {
1118 self.rich_compare(other, CompareOp::Eq)
1119 .and_then(|any| any.is_truthy())
1120 }
1121
1122 fn ne<O>(&self, other: O) -> PyResult<bool>
1123 where
1124 O: IntoPyObject<'py>,
1125 {
1126 self.rich_compare(other, CompareOp::Ne)
1127 .and_then(|any| any.is_truthy())
1128 }
1129
1130 fn gt<O>(&self, other: O) -> PyResult<bool>
1131 where
1132 O: IntoPyObject<'py>,
1133 {
1134 self.rich_compare(other, CompareOp::Gt)
1135 .and_then(|any| any.is_truthy())
1136 }
1137
1138 fn ge<O>(&self, other: O) -> PyResult<bool>
1139 where
1140 O: IntoPyObject<'py>,
1141 {
1142 self.rich_compare(other, CompareOp::Ge)
1143 .and_then(|any| any.is_truthy())
1144 }
1145
1146 implement_binop!(add, PyNumber_Add, "+");
1147 implement_binop!(sub, PyNumber_Subtract, "-");
1148 implement_binop!(mul, PyNumber_Multiply, "*");
1149 implement_binop!(matmul, PyNumber_MatrixMultiply, "@");
1150 implement_binop!(div, PyNumber_TrueDivide, "/");
1151 implement_binop!(floor_div, PyNumber_FloorDivide, "//");
1152 implement_binop!(rem, PyNumber_Remainder, "%");
1153 implement_binop!(lshift, PyNumber_Lshift, "<<");
1154 implement_binop!(rshift, PyNumber_Rshift, ">>");
1155 implement_binop!(bitand, PyNumber_And, "&");
1156 implement_binop!(bitor, PyNumber_Or, "|");
1157 implement_binop!(bitxor, PyNumber_Xor, "^");
1158
1159 fn divmod<O>(&self, other: O) -> PyResult<Bound<'py, PyAny>>
1161 where
1162 O: IntoPyObject<'py>,
1163 {
1164 fn inner<'py>(
1165 any: &Bound<'py, PyAny>,
1166 other: Borrowed<'_, 'py, PyAny>,
1167 ) -> PyResult<Bound<'py, PyAny>> {
1168 unsafe {
1169 ffi::PyNumber_Divmod(any.as_ptr(), other.as_ptr()).assume_owned_or_err(any.py())
1170 }
1171 }
1172
1173 let py = self.py();
1174 inner(
1175 self,
1176 other.into_pyobject_or_pyerr(py)?.into_any().as_borrowed(),
1177 )
1178 }
1179
1180 fn pow<O1, O2>(&self, other: O1, modulus: O2) -> PyResult<Bound<'py, PyAny>>
1183 where
1184 O1: IntoPyObject<'py>,
1185 O2: IntoPyObject<'py>,
1186 {
1187 fn inner<'py>(
1188 any: &Bound<'py, PyAny>,
1189 other: Borrowed<'_, 'py, PyAny>,
1190 modulus: Borrowed<'_, 'py, PyAny>,
1191 ) -> PyResult<Bound<'py, PyAny>> {
1192 unsafe {
1193 ffi::PyNumber_Power(any.as_ptr(), other.as_ptr(), modulus.as_ptr())
1194 .assume_owned_or_err(any.py())
1195 }
1196 }
1197
1198 let py = self.py();
1199 inner(
1200 self,
1201 other.into_pyobject_or_pyerr(py)?.into_any().as_borrowed(),
1202 modulus.into_pyobject_or_pyerr(py)?.into_any().as_borrowed(),
1203 )
1204 }
1205
1206 fn is_callable(&self) -> bool {
1207 unsafe { ffi::PyCallable_Check(self.as_ptr()) != 0 }
1208 }
1209
1210 fn call<A>(&self, args: A, kwargs: Option<&Bound<'py, PyDict>>) -> PyResult<Bound<'py, PyAny>>
1211 where
1212 A: IntoPyObject<'py, Target = PyTuple>,
1213 {
1214 fn inner<'py>(
1215 any: &Bound<'py, PyAny>,
1216 args: Borrowed<'_, 'py, PyTuple>,
1217 kwargs: Option<&Bound<'py, PyDict>>,
1218 ) -> PyResult<Bound<'py, PyAny>> {
1219 unsafe {
1220 ffi::PyObject_Call(
1221 any.as_ptr(),
1222 args.as_ptr(),
1223 kwargs.map_or(std::ptr::null_mut(), |dict| dict.as_ptr()),
1224 )
1225 .assume_owned_or_err(any.py())
1226 }
1227 }
1228
1229 let py = self.py();
1230 inner(self, args.into_pyobject_or_pyerr(py)?.as_borrowed(), kwargs)
1231 }
1232
1233 #[inline]
1234 fn call0(&self) -> PyResult<Bound<'py, PyAny>> {
1235 unsafe { ffi::compat::PyObject_CallNoArgs(self.as_ptr()).assume_owned_or_err(self.py()) }
1236 }
1237
1238 fn call1<A>(&self, args: A) -> PyResult<Bound<'py, PyAny>>
1239 where
1240 A: IntoPyObject<'py, Target = PyTuple>,
1241 {
1242 self.call(args, None)
1243 }
1244
1245 #[inline]
1246 fn call_method<N, A>(
1247 &self,
1248 name: N,
1249 args: A,
1250 kwargs: Option<&Bound<'py, PyDict>>,
1251 ) -> PyResult<Bound<'py, PyAny>>
1252 where
1253 N: IntoPyObject<'py, Target = PyString>,
1254 A: IntoPyObject<'py, Target = PyTuple>,
1255 {
1256 self.getattr(name)
1257 .and_then(|method| method.call(args, kwargs))
1258 }
1259
1260 #[inline]
1261 fn call_method0<N>(&self, name: N) -> PyResult<Bound<'py, PyAny>>
1262 where
1263 N: IntoPyObject<'py, Target = PyString>,
1264 {
1265 let py = self.py();
1266 let name = name.into_pyobject_or_pyerr(py)?.into_bound();
1267 unsafe {
1268 ffi::compat::PyObject_CallMethodNoArgs(self.as_ptr(), name.as_ptr())
1269 .assume_owned_or_err(py)
1270 }
1271 }
1272
1273 fn call_method1<N, A>(&self, name: N, args: A) -> PyResult<Bound<'py, PyAny>>
1274 where
1275 N: IntoPyObject<'py, Target = PyString>,
1276 A: IntoPyObject<'py, Target = PyTuple>,
1277 {
1278 self.call_method(name, args, None)
1279 }
1280
1281 fn is_truthy(&self) -> PyResult<bool> {
1282 let v = unsafe { ffi::PyObject_IsTrue(self.as_ptr()) };
1283 err::error_on_minusone(self.py(), v)?;
1284 Ok(v != 0)
1285 }
1286
1287 #[inline]
1288 fn is_none(&self) -> bool {
1289 unsafe { ffi::Py_None() == self.as_ptr() }
1290 }
1291
1292 fn is_ellipsis(&self) -> bool {
1293 unsafe { ffi::Py_Ellipsis() == self.as_ptr() }
1294 }
1295
1296 fn is_empty(&self) -> PyResult<bool> {
1297 self.len().map(|l| l == 0)
1298 }
1299
1300 fn get_item<K>(&self, key: K) -> PyResult<Bound<'py, PyAny>>
1301 where
1302 K: IntoPyObject<'py>,
1303 {
1304 fn inner<'py>(
1305 any: &Bound<'py, PyAny>,
1306 key: Borrowed<'_, 'py, PyAny>,
1307 ) -> PyResult<Bound<'py, PyAny>> {
1308 unsafe {
1309 ffi::PyObject_GetItem(any.as_ptr(), key.as_ptr()).assume_owned_or_err(any.py())
1310 }
1311 }
1312
1313 let py = self.py();
1314 inner(
1315 self,
1316 key.into_pyobject_or_pyerr(py)?.into_any().as_borrowed(),
1317 )
1318 }
1319
1320 fn set_item<K, V>(&self, key: K, value: V) -> PyResult<()>
1321 where
1322 K: IntoPyObject<'py>,
1323 V: IntoPyObject<'py>,
1324 {
1325 fn inner(
1326 any: &Bound<'_, PyAny>,
1327 key: Borrowed<'_, '_, PyAny>,
1328 value: Borrowed<'_, '_, PyAny>,
1329 ) -> PyResult<()> {
1330 err::error_on_minusone(any.py(), unsafe {
1331 ffi::PyObject_SetItem(any.as_ptr(), key.as_ptr(), value.as_ptr())
1332 })
1333 }
1334
1335 let py = self.py();
1336 inner(
1337 self,
1338 key.into_pyobject_or_pyerr(py)?.into_any().as_borrowed(),
1339 value.into_pyobject_or_pyerr(py)?.into_any().as_borrowed(),
1340 )
1341 }
1342
1343 fn del_item<K>(&self, key: K) -> PyResult<()>
1344 where
1345 K: IntoPyObject<'py>,
1346 {
1347 fn inner(any: &Bound<'_, PyAny>, key: Borrowed<'_, '_, PyAny>) -> PyResult<()> {
1348 err::error_on_minusone(any.py(), unsafe {
1349 ffi::PyObject_DelItem(any.as_ptr(), key.as_ptr())
1350 })
1351 }
1352
1353 let py = self.py();
1354 inner(
1355 self,
1356 key.into_pyobject_or_pyerr(py)?.into_any().as_borrowed(),
1357 )
1358 }
1359
1360 fn try_iter(&self) -> PyResult<Bound<'py, PyIterator>> {
1361 PyIterator::from_object(self)
1362 }
1363
1364 fn iter(&self) -> PyResult<Bound<'py, PyIterator>> {
1365 self.try_iter()
1366 }
1367
1368 fn get_type(&self) -> Bound<'py, PyType> {
1369 unsafe { PyType::from_borrowed_type_ptr(self.py(), ffi::Py_TYPE(self.as_ptr())) }
1370 }
1371
1372 #[inline]
1373 fn get_type_ptr(&self) -> *mut ffi::PyTypeObject {
1374 unsafe { ffi::Py_TYPE(self.as_ptr()) }
1375 }
1376
1377 #[inline]
1378 fn downcast<T>(&self) -> Result<&Bound<'py, T>, DowncastError<'_, 'py>>
1379 where
1380 T: PyTypeCheck,
1381 {
1382 if T::type_check(self) {
1383 Ok(unsafe { self.downcast_unchecked() })
1385 } else {
1386 Err(DowncastError::new(self, T::NAME))
1387 }
1388 }
1389
1390 #[inline]
1391 fn downcast_into<T>(self) -> Result<Bound<'py, T>, DowncastIntoError<'py>>
1392 where
1393 T: PyTypeCheck,
1394 {
1395 if T::type_check(&self) {
1396 Ok(unsafe { self.downcast_into_unchecked() })
1398 } else {
1399 Err(DowncastIntoError::new(self, T::NAME))
1400 }
1401 }
1402
1403 #[inline]
1404 fn downcast_exact<T>(&self) -> Result<&Bound<'py, T>, DowncastError<'_, 'py>>
1405 where
1406 T: PyTypeInfo,
1407 {
1408 if self.is_exact_instance_of::<T>() {
1409 Ok(unsafe { self.downcast_unchecked() })
1411 } else {
1412 Err(DowncastError::new(self, T::NAME))
1413 }
1414 }
1415
1416 #[inline]
1417 fn downcast_into_exact<T>(self) -> Result<Bound<'py, T>, DowncastIntoError<'py>>
1418 where
1419 T: PyTypeInfo,
1420 {
1421 if self.is_exact_instance_of::<T>() {
1422 Ok(unsafe { self.downcast_into_unchecked() })
1424 } else {
1425 Err(DowncastIntoError::new(self, T::NAME))
1426 }
1427 }
1428
1429 #[inline]
1430 unsafe fn downcast_unchecked<T>(&self) -> &Bound<'py, T> {
1431 &*ptr_from_ref(self).cast()
1432 }
1433
1434 #[inline]
1435 unsafe fn downcast_into_unchecked<T>(self) -> Bound<'py, T> {
1436 std::mem::transmute(self)
1437 }
1438
1439 fn extract<'a, T>(&'a self) -> PyResult<T>
1440 where
1441 T: FromPyObjectBound<'a, 'py>,
1442 {
1443 FromPyObjectBound::from_py_object_bound(self.as_borrowed())
1444 }
1445
1446 fn get_refcnt(&self) -> isize {
1447 unsafe { ffi::Py_REFCNT(self.as_ptr()) }
1448 }
1449
1450 fn repr(&self) -> PyResult<Bound<'py, PyString>> {
1451 unsafe {
1452 ffi::PyObject_Repr(self.as_ptr())
1453 .assume_owned_or_err(self.py())
1454 .downcast_into_unchecked()
1455 }
1456 }
1457
1458 fn str(&self) -> PyResult<Bound<'py, PyString>> {
1459 unsafe {
1460 ffi::PyObject_Str(self.as_ptr())
1461 .assume_owned_or_err(self.py())
1462 .downcast_into_unchecked()
1463 }
1464 }
1465
1466 fn hash(&self) -> PyResult<isize> {
1467 let v = unsafe { ffi::PyObject_Hash(self.as_ptr()) };
1468 crate::err::error_on_minusone(self.py(), v)?;
1469 Ok(v)
1470 }
1471
1472 fn len(&self) -> PyResult<usize> {
1473 let v = unsafe { ffi::PyObject_Size(self.as_ptr()) };
1474 crate::err::error_on_minusone(self.py(), v)?;
1475 Ok(v as usize)
1476 }
1477
1478 fn dir(&self) -> PyResult<Bound<'py, PyList>> {
1479 unsafe {
1480 ffi::PyObject_Dir(self.as_ptr())
1481 .assume_owned_or_err(self.py())
1482 .downcast_into_unchecked()
1483 }
1484 }
1485
1486 #[inline]
1487 fn is_instance(&self, ty: &Bound<'py, PyAny>) -> PyResult<bool> {
1488 let result = unsafe { ffi::PyObject_IsInstance(self.as_ptr(), ty.as_ptr()) };
1489 err::error_on_minusone(self.py(), result)?;
1490 Ok(result == 1)
1491 }
1492
1493 #[inline]
1494 fn is_exact_instance(&self, ty: &Bound<'py, PyAny>) -> bool {
1495 self.get_type().is(ty)
1496 }
1497
1498 #[inline]
1499 fn is_instance_of<T: PyTypeInfo>(&self) -> bool {
1500 T::is_type_of(self)
1501 }
1502
1503 #[inline]
1504 fn is_exact_instance_of<T: PyTypeInfo>(&self) -> bool {
1505 T::is_exact_type_of(self)
1506 }
1507
1508 fn contains<V>(&self, value: V) -> PyResult<bool>
1509 where
1510 V: IntoPyObject<'py>,
1511 {
1512 fn inner(any: &Bound<'_, PyAny>, value: Borrowed<'_, '_, PyAny>) -> PyResult<bool> {
1513 match unsafe { ffi::PySequence_Contains(any.as_ptr(), value.as_ptr()) } {
1514 0 => Ok(false),
1515 1 => Ok(true),
1516 _ => Err(PyErr::fetch(any.py())),
1517 }
1518 }
1519
1520 let py = self.py();
1521 inner(
1522 self,
1523 value.into_pyobject_or_pyerr(py)?.into_any().as_borrowed(),
1524 )
1525 }
1526
1527 #[cfg(not(any(PyPy, GraalPy)))]
1528 fn py_super(&self) -> PyResult<Bound<'py, PySuper>> {
1529 PySuper::new(&self.get_type(), self)
1530 }
1531}
1532
1533impl<'py> Bound<'py, PyAny> {
1534 #[allow(dead_code)] pub(crate) fn lookup_special<N>(&self, attr_name: N) -> PyResult<Option<Bound<'py, PyAny>>>
1546 where
1547 N: IntoPyObject<'py, Target = PyString>,
1548 {
1549 let py = self.py();
1550 let self_type = self.get_type();
1551 let attr = if let Ok(attr) = self_type.getattr(attr_name) {
1552 attr
1553 } else {
1554 return Ok(None);
1555 };
1556
1557 if let Some(descr_get) = attr.get_type().get_slot(TP_DESCR_GET) {
1559 unsafe {
1561 descr_get(attr.as_ptr(), self.as_ptr(), self_type.as_ptr())
1562 .assume_owned_or_err(py)
1563 .map(Some)
1564 }
1565 } else {
1566 Ok(Some(attr))
1567 }
1568 }
1569}
1570
1571#[cfg(test)]
1572mod tests {
1573 use crate::{
1574 basic::CompareOp,
1575 ffi,
1576 tests::common::generate_unique_module_name,
1577 types::{IntoPyDict, PyAny, PyAnyMethods, PyBool, PyInt, PyList, PyModule, PyTypeMethods},
1578 Bound, BoundObject, IntoPyObject, PyTypeInfo, Python,
1579 };
1580 use pyo3_ffi::c_str;
1581 use std::fmt::Debug;
1582
1583 #[test]
1584 fn test_lookup_special() {
1585 Python::with_gil(|py| {
1586 let module = PyModule::from_code(
1587 py,
1588 c_str!(
1589 r#"
1590class CustomCallable:
1591 def __call__(self):
1592 return 1
1593
1594class SimpleInt:
1595 def __int__(self):
1596 return 1
1597
1598class InheritedInt(SimpleInt): pass
1599
1600class NoInt: pass
1601
1602class NoDescriptorInt:
1603 __int__ = CustomCallable()
1604
1605class InstanceOverrideInt:
1606 def __int__(self):
1607 return 1
1608instance_override = InstanceOverrideInt()
1609instance_override.__int__ = lambda self: 2
1610
1611class ErrorInDescriptorInt:
1612 @property
1613 def __int__(self):
1614 raise ValueError("uh-oh!")
1615
1616class NonHeapNonDescriptorInt:
1617 # A static-typed callable that doesn't implement `__get__`. These are pretty hard to come by.
1618 __int__ = int
1619 "#
1620 ),
1621 c_str!("test.py"),
1622 &generate_unique_module_name("test"),
1623 )
1624 .unwrap();
1625
1626 let int = crate::intern!(py, "__int__");
1627 let eval_int =
1628 |obj: Bound<'_, PyAny>| obj.lookup_special(int)?.unwrap().call0()?.extract::<u32>();
1629
1630 let simple = module.getattr("SimpleInt").unwrap().call0().unwrap();
1631 assert_eq!(eval_int(simple).unwrap(), 1);
1632 let inherited = module.getattr("InheritedInt").unwrap().call0().unwrap();
1633 assert_eq!(eval_int(inherited).unwrap(), 1);
1634 let no_descriptor = module.getattr("NoDescriptorInt").unwrap().call0().unwrap();
1635 assert_eq!(eval_int(no_descriptor).unwrap(), 1);
1636 let missing = module.getattr("NoInt").unwrap().call0().unwrap();
1637 assert!(missing.lookup_special(int).unwrap().is_none());
1638 let instance_override = module.getattr("instance_override").unwrap();
1641 assert_eq!(eval_int(instance_override).unwrap(), 1);
1642 let descriptor_error = module
1643 .getattr("ErrorInDescriptorInt")
1644 .unwrap()
1645 .call0()
1646 .unwrap();
1647 assert!(descriptor_error.lookup_special(int).is_err());
1648 let nonheap_nondescriptor = module
1649 .getattr("NonHeapNonDescriptorInt")
1650 .unwrap()
1651 .call0()
1652 .unwrap();
1653 assert_eq!(eval_int(nonheap_nondescriptor).unwrap(), 0);
1654 })
1655 }
1656
1657 #[test]
1658 fn test_call_for_non_existing_method() {
1659 Python::with_gil(|py| {
1660 let a = py.eval(ffi::c_str!("42"), None, None).unwrap();
1661 a.call_method0("__str__").unwrap(); assert!(a.call_method("nonexistent_method", (1,), None).is_err());
1663 assert!(a.call_method0("nonexistent_method").is_err());
1664 assert!(a.call_method1("nonexistent_method", (1,)).is_err());
1665 });
1666 }
1667
1668 #[test]
1669 fn test_call_with_kwargs() {
1670 Python::with_gil(|py| {
1671 let list = vec![3, 6, 5, 4, 7].into_pyobject(py).unwrap();
1672 let dict = vec![("reverse", true)].into_py_dict(py).unwrap();
1673 list.call_method("sort", (), Some(&dict)).unwrap();
1674 assert_eq!(list.extract::<Vec<i32>>().unwrap(), vec![7, 6, 5, 4, 3]);
1675 });
1676 }
1677
1678 #[test]
1679 fn test_call_method0() {
1680 Python::with_gil(|py| {
1681 let module = PyModule::from_code(
1682 py,
1683 c_str!(
1684 r#"
1685class SimpleClass:
1686 def foo(self):
1687 return 42
1688"#
1689 ),
1690 c_str!(file!()),
1691 &generate_unique_module_name("test_module"),
1692 )
1693 .expect("module creation failed");
1694
1695 let simple_class = module.getattr("SimpleClass").unwrap().call0().unwrap();
1696 assert_eq!(
1697 simple_class
1698 .call_method0("foo")
1699 .unwrap()
1700 .extract::<u32>()
1701 .unwrap(),
1702 42
1703 );
1704 })
1705 }
1706
1707 #[test]
1708 fn test_type() {
1709 Python::with_gil(|py| {
1710 let obj = py.eval(ffi::c_str!("42"), None, None).unwrap();
1711 assert_eq!(obj.get_type().as_type_ptr(), obj.get_type_ptr());
1712 });
1713 }
1714
1715 #[test]
1716 fn test_dir() {
1717 Python::with_gil(|py| {
1718 let obj = py.eval(ffi::c_str!("42"), None, None).unwrap();
1719 let dir = py
1720 .eval(ffi::c_str!("dir(42)"), None, None)
1721 .unwrap()
1722 .downcast_into::<PyList>()
1723 .unwrap();
1724 let a = obj
1725 .dir()
1726 .unwrap()
1727 .into_iter()
1728 .map(|x| x.extract::<String>().unwrap());
1729 let b = dir.into_iter().map(|x| x.extract::<String>().unwrap());
1730 assert!(a.eq(b));
1731 });
1732 }
1733
1734 #[test]
1735 fn test_hasattr() {
1736 Python::with_gil(|py| {
1737 let x = 5i32.into_pyobject(py).unwrap();
1738 assert!(x.is_instance_of::<PyInt>());
1739
1740 assert!(x.hasattr("to_bytes").unwrap());
1741 assert!(!x.hasattr("bbbbbbytes").unwrap());
1742 })
1743 }
1744
1745 #[cfg(feature = "macros")]
1746 #[test]
1747 #[allow(unknown_lints, non_local_definitions)]
1748 fn test_hasattr_error() {
1749 use crate::exceptions::PyValueError;
1750 use crate::prelude::*;
1751
1752 #[pyclass(crate = "crate")]
1753 struct GetattrFail;
1754
1755 #[pymethods(crate = "crate")]
1756 impl GetattrFail {
1757 fn __getattr__(&self, attr: PyObject) -> PyResult<PyObject> {
1758 Err(PyValueError::new_err(attr))
1759 }
1760 }
1761
1762 Python::with_gil(|py| {
1763 let obj = Py::new(py, GetattrFail).unwrap();
1764 let obj = obj.bind(py).as_ref();
1765
1766 assert!(obj
1767 .hasattr("foo")
1768 .unwrap_err()
1769 .is_instance_of::<PyValueError>(py));
1770 })
1771 }
1772
1773 #[test]
1774 fn test_nan_eq() {
1775 Python::with_gil(|py| {
1776 let nan = py.eval(ffi::c_str!("float('nan')"), None, None).unwrap();
1777 assert!(nan.compare(&nan).is_err());
1778 });
1779 }
1780
1781 #[test]
1782 fn test_any_is_instance_of() {
1783 Python::with_gil(|py| {
1784 let x = 5i32.into_pyobject(py).unwrap();
1785 assert!(x.is_instance_of::<PyInt>());
1786
1787 let l = vec![&x, &x].into_pyobject(py).unwrap();
1788 assert!(l.is_instance_of::<PyList>());
1789 });
1790 }
1791
1792 #[test]
1793 fn test_any_is_instance() {
1794 Python::with_gil(|py| {
1795 let l = vec![1i8, 2].into_pyobject(py).unwrap();
1796 assert!(l.is_instance(&py.get_type::<PyList>()).unwrap());
1797 });
1798 }
1799
1800 #[test]
1801 fn test_any_is_exact_instance_of() {
1802 Python::with_gil(|py| {
1803 let x = 5i32.into_pyobject(py).unwrap();
1804 assert!(x.is_exact_instance_of::<PyInt>());
1805
1806 let t = PyBool::new(py, true);
1807 assert!(t.is_instance_of::<PyInt>());
1808 assert!(!t.is_exact_instance_of::<PyInt>());
1809 assert!(t.is_exact_instance_of::<PyBool>());
1810
1811 let l = vec![&x, &x].into_pyobject(py).unwrap();
1812 assert!(l.is_exact_instance_of::<PyList>());
1813 });
1814 }
1815
1816 #[test]
1817 fn test_any_is_exact_instance() {
1818 Python::with_gil(|py| {
1819 let t = PyBool::new(py, true);
1820 assert!(t.is_instance(&py.get_type::<PyInt>()).unwrap());
1821 assert!(!t.is_exact_instance(&py.get_type::<PyInt>()));
1822 assert!(t.is_exact_instance(&py.get_type::<PyBool>()));
1823 });
1824 }
1825
1826 #[test]
1827 fn test_any_contains() {
1828 Python::with_gil(|py| {
1829 let v: Vec<i32> = vec![1, 1, 2, 3, 5, 8];
1830 let ob = v.into_pyobject(py).unwrap();
1831
1832 let bad_needle = 7i32.into_pyobject(py).unwrap();
1833 assert!(!ob.contains(&bad_needle).unwrap());
1834
1835 let good_needle = 8i32.into_pyobject(py).unwrap();
1836 assert!(ob.contains(&good_needle).unwrap());
1837
1838 let type_coerced_needle = 8f32.into_pyobject(py).unwrap();
1839 assert!(ob.contains(&type_coerced_needle).unwrap());
1840
1841 let n: u32 = 42;
1842 let bad_haystack = n.into_pyobject(py).unwrap();
1843 let irrelevant_needle = 0i32.into_pyobject(py).unwrap();
1844 assert!(bad_haystack.contains(&irrelevant_needle).is_err());
1845 });
1846 }
1847
1848 fn test_eq_methods_generic<'a, T>(list: &'a [T])
1850 where
1851 T: PartialEq + PartialOrd,
1852 for<'py> &'a T: IntoPyObject<'py>,
1853 for<'py> <&'a T as IntoPyObject<'py>>::Error: Debug,
1854 {
1855 Python::with_gil(|py| {
1856 for a in list {
1857 for b in list {
1858 let a_py = a.into_pyobject(py).unwrap().into_any().into_bound();
1859 let b_py = b.into_pyobject(py).unwrap().into_any().into_bound();
1860
1861 assert_eq!(
1862 a.lt(b),
1863 a_py.lt(&b_py).unwrap(),
1864 "{} < {} should be {}.",
1865 a_py,
1866 b_py,
1867 a.lt(b)
1868 );
1869 assert_eq!(
1870 a.le(b),
1871 a_py.le(&b_py).unwrap(),
1872 "{} <= {} should be {}.",
1873 a_py,
1874 b_py,
1875 a.le(b)
1876 );
1877 assert_eq!(
1878 a.eq(b),
1879 a_py.eq(&b_py).unwrap(),
1880 "{} == {} should be {}.",
1881 a_py,
1882 b_py,
1883 a.eq(b)
1884 );
1885 assert_eq!(
1886 a.ne(b),
1887 a_py.ne(&b_py).unwrap(),
1888 "{} != {} should be {}.",
1889 a_py,
1890 b_py,
1891 a.ne(b)
1892 );
1893 assert_eq!(
1894 a.gt(b),
1895 a_py.gt(&b_py).unwrap(),
1896 "{} > {} should be {}.",
1897 a_py,
1898 b_py,
1899 a.gt(b)
1900 );
1901 assert_eq!(
1902 a.ge(b),
1903 a_py.ge(&b_py).unwrap(),
1904 "{} >= {} should be {}.",
1905 a_py,
1906 b_py,
1907 a.ge(b)
1908 );
1909 }
1910 }
1911 });
1912 }
1913
1914 #[test]
1915 fn test_eq_methods_integers() {
1916 let ints = [-4, -4, 1, 2, 0, -100, 1_000_000];
1917 test_eq_methods_generic::<i32>(&ints);
1918 }
1919
1920 #[test]
1921 fn test_eq_methods_strings() {
1922 let strings = ["Let's", "test", "some", "eq", "methods"];
1923 test_eq_methods_generic::<&str>(&strings);
1924 }
1925
1926 #[test]
1927 fn test_eq_methods_floats() {
1928 let floats = [
1929 -1.0,
1930 2.5,
1931 0.0,
1932 3.0,
1933 std::f64::consts::PI,
1934 10.0,
1935 10.0 / 3.0,
1936 -1_000_000.0,
1937 ];
1938 test_eq_methods_generic::<f64>(&floats);
1939 }
1940
1941 #[test]
1942 fn test_eq_methods_bools() {
1943 let bools = [true, false];
1944 test_eq_methods_generic::<bool>(&bools);
1945 }
1946
1947 #[test]
1948 fn test_rich_compare_type_error() {
1949 Python::with_gil(|py| {
1950 let py_int = 1i32.into_pyobject(py).unwrap();
1951 let py_str = "1".into_pyobject(py).unwrap();
1952
1953 assert!(py_int.rich_compare(&py_str, CompareOp::Lt).is_err());
1954 assert!(!py_int
1955 .rich_compare(py_str, CompareOp::Eq)
1956 .unwrap()
1957 .is_truthy()
1958 .unwrap());
1959 })
1960 }
1961
1962 #[test]
1963 #[allow(deprecated)]
1964 fn test_is_ellipsis() {
1965 Python::with_gil(|py| {
1966 let v = py
1967 .eval(ffi::c_str!("..."), None, None)
1968 .map_err(|e| e.display(py))
1969 .unwrap();
1970
1971 assert!(v.is_ellipsis());
1972
1973 let not_ellipsis = 5i32.into_pyobject(py).unwrap();
1974 assert!(!not_ellipsis.is_ellipsis());
1975 });
1976 }
1977
1978 #[test]
1979 fn test_is_callable() {
1980 Python::with_gil(|py| {
1981 assert!(PyList::type_object(py).is_callable());
1982
1983 let not_callable = 5i32.into_pyobject(py).unwrap();
1984 assert!(!not_callable.is_callable());
1985 });
1986 }
1987
1988 #[test]
1989 fn test_is_empty() {
1990 Python::with_gil(|py| {
1991 let empty_list = PyList::empty(py).into_any();
1992 assert!(empty_list.is_empty().unwrap());
1993
1994 let list = PyList::new(py, vec![1, 2, 3]).unwrap().into_any();
1995 assert!(!list.is_empty().unwrap());
1996
1997 let not_container = 5i32.into_pyobject(py).unwrap();
1998 assert!(not_container.is_empty().is_err());
1999 });
2000 }
2001
2002 #[cfg(feature = "macros")]
2003 #[test]
2004 #[allow(unknown_lints, non_local_definitions)]
2005 fn test_fallible_dir() {
2006 use crate::exceptions::PyValueError;
2007 use crate::prelude::*;
2008
2009 #[pyclass(crate = "crate")]
2010 struct DirFail;
2011
2012 #[pymethods(crate = "crate")]
2013 impl DirFail {
2014 fn __dir__(&self) -> PyResult<PyObject> {
2015 Err(PyValueError::new_err("uh-oh!"))
2016 }
2017 }
2018
2019 Python::with_gil(|py| {
2020 let obj = Bound::new(py, DirFail).unwrap();
2021 assert!(obj.dir().unwrap_err().is_instance_of::<PyValueError>(py));
2022 })
2023 }
2024}