1#![cfg(feature = "either")]
23//! Conversion to/from
4//! [either](https://docs.rs/either/ "A library for easy idiomatic error handling and reporting in Rust applications")’s
5//! [`Either`] type to a union of two Python types.
6//!
7//! Use of a generic sum type like [either] is common when you want to either accept one of two possible
8//! types as an argument or return one of two possible types from a function, without having to define
9//! a helper type manually yourself.
10//!
11//! # Setup
12//!
13//! To use this feature, add this to your **`Cargo.toml`**:
14//!
15//! ```toml
16//! [dependencies]
17//! ## change * to the version you want to use, ideally the latest.
18//! either = "*"
19#![doc = concat!("pyo3 = { version = \"", env!("CARGO_PKG_VERSION"), "\", features = [\"either\"] }")]
20//! ```
21//!
22//! Note that you must use compatible versions of either and PyO3.
23//! The required either version may vary based on the version of PyO3.
24//!
25//! # Example: Convert a `int | str` to `Either<i32, String>`.
26//!
27//! ```rust
28//! use either::Either;
29//! use pyo3::{Python, PyResult, IntoPyObject, types::PyAnyMethods};
30//!
31//! fn main() -> PyResult<()> {
32//! pyo3::prepare_freethreaded_python();
33//! Python::with_gil(|py| {
34//! // Create a string and an int in Python.
35//! let py_str = "crab".into_pyobject(py)?;
36//! let py_int = 42i32.into_pyobject(py)?;
37//! // Now convert it to an Either<i32, String>.
38//! let either_str: Either<i32, String> = py_str.extract()?;
39//! let either_int: Either<i32, String> = py_int.extract()?;
40//! Ok(())
41//! })
42//! }
43//! ```
44//!
45//! [either](https://docs.rs/either/ "A library for easy idiomatic error handling and reporting in Rust applications")’s
4647#[cfg(feature = "experimental-inspect")]
48use crate::inspect::types::TypeInfo;
49use crate::{
50 exceptions::PyTypeError, types::any::PyAnyMethods, Bound, FromPyObject, IntoPyObject,
51 IntoPyObjectExt, PyAny, PyErr, PyResult, Python,
52};
53use either::Either;
5455#[cfg_attr(docsrs, doc(cfg(feature = "either")))]
56impl<'py, L, R> IntoPyObject<'py> for Either<L, R>
57where
58L: IntoPyObject<'py>,
59 R: IntoPyObject<'py>,
60{
61type Target = PyAny;
62type Output = Bound<'py, Self::Target>;
63type Error = PyErr;
6465fn into_pyobject(self, py: Python<'py>) -> Result<Self::Output, Self::Error> {
66match self {
67 Either::Left(l) => l.into_bound_py_any(py),
68 Either::Right(r) => r.into_bound_py_any(py),
69 }
70 }
71}
7273#[cfg_attr(docsrs, doc(cfg(feature = "either")))]
74impl<'a, 'py, L, R> IntoPyObject<'py> for &'a Either<L, R>
75where
76&'a L: IntoPyObject<'py>,
77&'a R: IntoPyObject<'py>,
78{
79type Target = PyAny;
80type Output = Bound<'py, Self::Target>;
81type Error = PyErr;
8283fn into_pyobject(self, py: Python<'py>) -> Result<Self::Output, Self::Error> {
84match self {
85 Either::Left(l) => l.into_bound_py_any(py),
86 Either::Right(r) => r.into_bound_py_any(py),
87 }
88 }
89}
9091#[cfg_attr(docsrs, doc(cfg(feature = "either")))]
92impl<'py, L, R> FromPyObject<'py> for Either<L, R>
93where
94L: FromPyObject<'py>,
95 R: FromPyObject<'py>,
96{
97#[inline]
98fn extract_bound(obj: &Bound<'py, PyAny>) -> PyResult<Self> {
99if let Ok(l) = obj.extract::<L>() {
100Ok(Either::Left(l))
101 } else if let Ok(r) = obj.extract::<R>() {
102Ok(Either::Right(r))
103 } else {
104// TODO: it might be nice to use the `type_input()` name here once `type_input`
105 // is not experimental, rather than the Rust type names.
106let err_msg = format!(
107"failed to convert the value to 'Union[{}, {}]'",
108 std::any::type_name::<L>(),
109 std::any::type_name::<R>()
110 );
111Err(PyTypeError::new_err(err_msg))
112 }
113 }
114115#[cfg(feature = "experimental-inspect")]
116fn type_input() -> TypeInfo {
117 TypeInfo::union_of(&[L::type_input(), R::type_input()])
118 }
119}
120121#[cfg(test)]
122mod tests {
123use std::borrow::Cow;
124125use crate::exceptions::PyTypeError;
126use crate::{IntoPyObject, Python};
127128use crate::types::PyAnyMethods;
129use either::Either;
130131#[test]
132fn test_either_conversion() {
133type E = Either<i32, String>;
134type E1 = Either<i32, f32>;
135type E2 = Either<f32, i32>;
136137 Python::with_gil(|py| {
138let l = E::Left(42);
139let obj_l = (&l).into_pyobject(py).unwrap();
140assert_eq!(obj_l.extract::<i32>().unwrap(), 42);
141assert_eq!(obj_l.extract::<E>().unwrap(), l);
142143let r = E::Right("foo".to_owned());
144let obj_r = (&r).into_pyobject(py).unwrap();
145assert_eq!(obj_r.extract::<Cow<'_, str>>().unwrap(), "foo");
146assert_eq!(obj_r.extract::<E>().unwrap(), r);
147148let obj_s = "foo".into_pyobject(py).unwrap();
149let err = obj_s.extract::<E1>().unwrap_err();
150assert!(err.is_instance_of::<PyTypeError>(py));
151assert_eq!(
152 err.to_string(),
153"TypeError: failed to convert the value to 'Union[i32, f32]'"
154);
155156let obj_i = 42i32.into_pyobject(py).unwrap();
157assert_eq!(obj_i.extract::<E1>().unwrap(), E1::Left(42));
158assert_eq!(obj_i.extract::<E2>().unwrap(), E2::Left(42.0));
159160let obj_f = 42.0f64.into_pyobject(py).unwrap();
161assert_eq!(obj_f.extract::<E1>().unwrap(), E1::Right(42.0));
162assert_eq!(obj_f.extract::<E2>().unwrap(), E2::Left(42.0));
163 });
164 }
165}