Skip to content

Commit

Permalink
Avoids panic on class initialization when adding a class to the module
Browse files Browse the repository at this point in the history
  • Loading branch information
Tpt committed Feb 19, 2024
1 parent 0b7fdd4 commit f71feec
Show file tree
Hide file tree
Showing 5 changed files with 20 additions and 4 deletions.
10 changes: 10 additions & 0 deletions pyo3-macros-backend/src/pyclass.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1290,6 +1290,16 @@ fn impl_pytypeinfo(
.get_or_init(py)
.as_type_ptr()
}

#[inline]
fn try_type_object_bound(py: _pyo3::Python<'_>) -> _pyo3::PyResult<_pyo3::Bound<'_, _pyo3::types::PyType>> {
use ::std::borrow::ToOwned;
#deprecations

<#cls as _pyo3::impl_::pyclass::PyClassImpl>::lazy_type_object()
.get_or_try_init(py)
.map(|t| t.to_owned())
}
}
}
}
Expand Down
2 changes: 1 addition & 1 deletion src/impl_/pyclass/lazy_type_object.rs
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@ impl<T: PyClass> LazyTypeObject<T> {
}

/// Fallible version of the above.
pub(crate) fn get_or_try_init<'py>(&self, py: Python<'py>) -> PyResult<&Bound<'py, PyType>> {
pub fn get_or_try_init<'py>(&self, py: Python<'py>) -> PyResult<&Bound<'py, PyType>> {
self.0
.get_or_try_init(py, create_type_object::<T>, T::NAME, T::items_iter())
}
Expand Down
2 changes: 1 addition & 1 deletion src/impl_/pymodule.rs
Original file line number Diff line number Diff line change
Expand Up @@ -143,7 +143,7 @@ pub trait PyAddToModule {

impl<T: PyTypeInfo> PyAddToModule for T {
fn add_to_module(module: &Bound<'_, PyModule>) -> PyResult<()> {
module.add(Self::NAME, Self::type_object_bound(module.py()))
module.add(Self::NAME, Self::try_type_object_bound(module.py())?)
}
}

Expand Down
8 changes: 7 additions & 1 deletion src/type_object.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@
use crate::ffi_ptr_ext::FfiPtrExt;
use crate::types::any::PyAnyMethods;
use crate::types::{PyAny, PyType};
use crate::{ffi, Bound, PyNativeType, Python};
use crate::{ffi, Bound, PyNativeType, PyResult, Python};

/// `T: PyLayout<U>` represents that `T` is a concrete representation of `U` in the Python heap.
/// E.g., `PyCell` is a concrete representation of all `pyclass`es, and `ffi::PyObject`
Expand Down Expand Up @@ -96,6 +96,12 @@ pub unsafe trait PyTypeInfo: Sized + HasPyGilRef {
}
}

/// Returns the safe abstraction over the type object or an error if initialization fails
#[inline]
fn try_type_object_bound(py: Python<'_>) -> PyResult<Bound<'_, PyType>> {
Ok(Self::type_object_bound(py))
}

/// Checks if `object` is an instance of this type or a subclass of this type.
#[inline]
#[cfg_attr(
Expand Down
2 changes: 1 addition & 1 deletion src/types/module.rs
Original file line number Diff line number Diff line change
Expand Up @@ -633,7 +633,7 @@ impl<'py> PyModuleMethods<'py> for Bound<'py, PyModule> {
T: PyClass,
{
let py = self.py();
self.add(T::NAME, T::lazy_type_object().get_or_try_init(py)?)
self.add(T::NAME, T::try_type_object_bound(py)?)
}

fn add_wrapped<T>(&self, wrapper: &impl Fn(Python<'py>) -> T) -> PyResult<()>
Expand Down

0 comments on commit f71feec

Please sign in to comment.