guide/src/type-stub.md
*.pyi files) and introspectionThis feature is still in active development. See the related issue.
For documentation on type stubs and how to use them with stable PyO3, refer to this page
PyO3 has a work in progress support to generate type stub files.
It works using:
#[pyclass]) that generate constant JSON strings that are then included in the built binaries by rustc if the experimental-inspect feature is enabled.pyo3-introspection crate that can parse the generated binaries, extract the JSON strings and build stub files from it.maturin exposing pyo3-introspection features in their CLI API.For example, the following Rust code
#[pymodule]
pub mod example {
use pyo3::prelude::*;
#[pymodule_export]
pub const CONSTANT: &str = "FOO";
#[pyclass(eq)]
#[derive(Eq)]
struct Class {
value: usize
}
#[pymethods]
impl Class {
#[new]
fn new(value: usize) -> Self {
Self { value }
}
#[getter]
fn value(&self) -> usize {
self.value
}
}
#[pyfunction]
#[pyo3(signature = (arg: "list[int]") -> "list[int]")]
fn list_of_int_identity(arg: Bound<'_, PyAny>) -> Bound<'_, PyAny> {
arg
}
}
will generate the following stub file:
import typing
CONSTANT: typing.Final = "FOO"
class Class:
def __init__(self, value: int) -> None: ...
@property
def value(self) -> int: ...
def __eq__(self, other: Class) -> bool: ...
def __ne__(self, other: Class) -> bool: ...
def list_of_int_identity(arg: list[int]) -> list[int]: ...
The only piece of added syntax is that the #[pyo3(signature = ...)] attribute can now contain type annotations like #[pyo3(signature = (arg: "list[int]") -> "list[int]")] (note the "" around type annotations).
This is useful when PyO3 is not able to derive proper type annotations by itself.
experimental-inspect feature is required to generate the introspection fragments.FromPyObject::INPUT_TYPE and IntoPyObject::OUTPUT_TYPE must be implemented for PyO3 to get the proper input/output type annotations to use.#[pymodule] and #[pymodule_init] functions.
If they are present, the module is tagged as incomplete using a fake def __getattr__(name: str) -> Incomplete: ... function following best practices.