pyo3 0.29.0

Bindings to Python interpreter
Documentation
#![deny(clippy::undocumented_unsafe_blocks)]
use crate::ffi_ptr_ext::FfiPtrExt;
use crate::sealed::Sealed;
use crate::types::{PyCode, PyDict};
use crate::PyAny;
use crate::{ffi, Bound, PyResult, Python};
use core::ffi::CStr;
use pyo3_ffi::PyObject;

/// Represents a Python frame.
///
/// Values of this type are accessed via PyO3's smart pointers, e.g. as
/// [`Py<PyFrame>`][crate::Py] or [`Bound<'py, PyFrame>`][crate::Bound].
#[repr(transparent)]
pub struct PyFrame(PyAny);

pyobject_native_type_core!(
    PyFrame,
    pyobject_native_static_type_object!(ffi::PyFrame_Type),
    "types",
    "FrameType",
    #checkfunction=ffi::PyFrame_Check
);

impl PyFrame {
    /// Creates a new frame object.
    pub fn new<'py>(
        py: Python<'py>,
        file_name: &CStr,
        func_name: &CStr,
        line_number: i32,
    ) -> PyResult<Bound<'py, PyFrame>> {
        // Safety: Thread is attached because we have a python token
        let state = unsafe { ffi::compat::PyThreadState_GetUnchecked() };
        let code = PyCode::empty(py, file_name, func_name, line_number);
        let globals = PyDict::new(py);
        let locals = PyDict::new(py);

        // SAFETY:
        // - we're attached to the interpreter
        // - `PyFrame_New` returns an owned reference or raises an exception
        // - the result is a frame object
        unsafe {
            Ok(ffi::PyFrame_New(
                state,
                code.as_ptr().cast(),
                globals.as_ptr(),
                locals.as_ptr(),
            )
            .cast::<PyObject>()
            .assume_owned_or_err(py)?
            .cast_into_unchecked::<PyFrame>())
        }
    }
}

/// Implementation of functionality for [`PyFrame`].
///
/// These methods are defined for the `Bound<'py, PyFrame>` smart pointer, so to use method call
/// syntax these methods are separated into a trait, because stable Rust does not yet support
/// `arbitrary_self_types`.
#[doc(alias = "PyFrame")]
pub trait PyFrameMethods<'py>: Sealed {
    /// Returns the line number of the current instruction in the frame.
    fn line_number(&self) -> i32;

    /// Gets this frame's next outer frame if there is one
    #[cfg(all(Py_3_9, not(Py_LIMITED_API)))]
    fn outer(&self) -> Option<Bound<'py, PyFrame>>;

    /// Gets the frame code
    #[cfg(any(all(Py_3_9, not(Py_LIMITED_API)), Py_3_10))]
    fn code(&self) -> Bound<'py, PyCode>;

    /// Gets the variable `name` of this frame.
    #[cfg(all(Py_3_12, not(Py_LIMITED_API)))]
    fn var(&self, name: &CStr) -> PyResult<Bound<'py, PyAny>>;

    /// Gets this frame's `f_builtins` attribute
    #[cfg(all(Py_3_11, not(Py_LIMITED_API)))]
    fn builtins(&self) -> Bound<'py, PyDict>;

    /// Gets this frame's `f_globals` attribute
    #[cfg(all(Py_3_11, not(Py_LIMITED_API)))]
    fn globals(&self) -> Bound<'py, PyDict>;

    /// Gets this frame's `f_locals` attribute
    #[cfg(all(Py_3_11, not(Py_LIMITED_API)))]
    fn locals(&self) -> Bound<'py, PyAny>;
}

impl<'py> PyFrameMethods<'py> for Bound<'py, PyFrame> {
    fn line_number(&self) -> i32 {
        // SAFETY:
        // - we're attached to the interpreter
        // - `self` is a `PyFrameObject`
        unsafe { ffi::PyFrame_GetLineNumber(self.as_ptr().cast()) }
    }

    #[cfg(all(Py_3_9, not(Py_LIMITED_API)))]
    fn outer(&self) -> Option<Bound<'py, PyFrame>> {
        // SAFETY:
        // - we're attached to the interpreter
        // - `self` is a `PyFrameObject`
        // - `PyFrame_GetBack` returns an owned reference
        // - the result may be null if there is no outer frame, but no exception is raised
        // - the result is a frame object
        unsafe {
            ffi::PyFrame_GetBack(self.as_ptr().cast())
                .cast::<ffi::PyObject>()
                .assume_owned_or_opt(self.py())
                .map(|obj| obj.cast_into_unchecked())
        }
    }

    #[cfg(any(all(Py_3_9, not(Py_LIMITED_API)), Py_3_10))]
    fn code(&self) -> Bound<'py, PyCode> {
        // SAFETY:
        // - we're attached to the interpreter
        // - `self` is a `PyFrameObject`
        // - `PyFrame_GetCode` returns an owned reference
        // - the result can not be null
        // - the result is a code object
        unsafe {
            ffi::PyFrame_GetCode(self.as_ptr().cast())
                .cast::<ffi::PyObject>()
                .assume_owned_unchecked(self.py())
                .cast_into_unchecked()
        }
    }

    #[cfg(all(Py_3_12, not(Py_LIMITED_API)))]
    fn var(&self, name: &CStr) -> PyResult<Bound<'py, PyAny>> {
        // SAFETY:
        // - we're attached to the interpreter
        // - `self` is a `PyFrameObject`
        // - `PyFrame_GetVarString` returns an owned reference or raises an exception
        // - `name` is a valid null terminated C string
        unsafe {
            ffi::PyFrame_GetVarString(self.as_ptr().cast(), name.as_ptr().cast_mut())
                .assume_owned_or_err(self.py())
        }
    }

    #[cfg(all(Py_3_11, not(Py_LIMITED_API)))]
    fn builtins(&self) -> Bound<'py, PyDict> {
        // SAFETY:
        // - we're attached to the interpreter
        // - `self` is a `PyFrameObject`
        // - `PyFrame_GetBuiltins` returns an owned reference
        // - the result can not be null
        unsafe {
            ffi::PyFrame_GetBuiltins(self.as_ptr().cast())
                .assume_owned_unchecked(self.py())
                .cast_into()
                // The result is expected (and documented) to be a dict object, however it is
                // possible for Python code to overwrite `__builtins__` with any arbitrary object.
                // As reasonable code should never do this, we panic here for correctness in case
                // the type does not match.
                //
                // See https://github.com/PyO3/pyo3/issues/6048
                .expect("`PyFrame_GetBuiltins` returns a `dict`")
        }
    }

    #[cfg(all(Py_3_11, not(Py_LIMITED_API)))]
    fn globals(&self) -> Bound<'py, PyDict> {
        // SAFETY:
        // - we're attached to the interpreter
        // - `self` is a `PyFrameObject`
        // - `PyFrame_GetGlobals` returns an owned reference
        // - the result can not be null
        // - the result is a dict object
        unsafe {
            ffi::PyFrame_GetGlobals(self.as_ptr().cast())
                .assume_owned_unchecked(self.py())
                .cast_into_unchecked()
        }
    }

    #[cfg(all(Py_3_11, not(Py_LIMITED_API)))]
    fn locals(&self) -> Bound<'py, PyAny> {
        // SAFETY:
        // - we're attached to the interpreter
        // - `self` is a `PyFrameObject`
        // - `PyFrame_GetLocals` returns an owned reference
        // - the result can not be null
        unsafe { ffi::PyFrame_GetLocals(self.as_ptr().cast()).assume_owned_unchecked(self.py()) }
    }
}

#[cfg(test)]
mod tests {
    use super::*;

    #[cfg(Py_3_9)]
    fn get_frame(py: Python<'_>) -> Bound<'_, PyFrame> {
        use crate::types::PyAnyMethods as _;

        let m = crate::types::PyModule::from_code(
            py,
            cr#"
import sys
CONST = "global"
def get_frame():
    var = 42
    return sys._getframe()
"#,
            c"frame.py",
            c"frame",
        )
        .unwrap();

        m.getattr("get_frame")
            .unwrap()
            .call0()
            .unwrap()
            .cast_into()
            .unwrap()
    }

    #[test]
    fn test_frame_creation() {
        Python::attach(|py| {
            let frame = PyFrame::new(py, c"file.py", c"func", 42).unwrap();
            assert_eq!(frame.line_number(), 42);
        });
    }

    #[test]
    #[cfg(all(Py_3_9, not(Py_LIMITED_API)))]
    fn test_frame_outer() {
        Python::attach(|py| {
            use crate::types::PyAnyMethods as _;

            let m = crate::types::PyModule::from_code(
                py,
                cr#"
import sys
def inner():
    return sys._getframe()
def outer():
    return inner()
"#,
                c"outer.py",
                c"outer",
            )
            .unwrap();

            let frame = m
                .getattr("outer")
                .unwrap()
                .call0()
                .unwrap()
                .cast_into()
                .unwrap();

            let back = frame.outer().unwrap();
            let f_back = frame.getattr("f_back").unwrap();

            assert_eq!(back.as_ptr(), f_back.as_ptr());
            assert_eq!(back.line_number(), 6)
        })
    }

    #[test]
    #[cfg(any(all(Py_3_9, not(Py_LIMITED_API)), Py_3_10))]
    fn test_frame_get_code() {
        Python::attach(|py| {
            use crate::types::PyAnyMethods as _;

            let frame = get_frame(py);
            let code = frame.code();
            let f_code = frame.getattr("f_code").unwrap();

            assert_eq!(code.as_ptr(), f_code.as_ptr());
        })
    }

    #[test]
    #[cfg(all(Py_3_12, not(Py_LIMITED_API)))]
    fn test_frame_get_var() {
        Python::attach(|py| {
            use crate::types::PyAnyMethods as _;

            let frame = get_frame(py);
            assert_eq!(frame.var(c"var").unwrap().extract::<u32>().unwrap(), 42);
            assert!(frame.var(c"var2").is_err());
        })
    }

    #[test]
    #[cfg(all(Py_3_11, not(Py_LIMITED_API)))]
    fn test_frame_get_builtins() {
        Python::attach(|py| {
            use crate::types::PyAnyMethods as _;

            let frame = get_frame(py);
            let builtins = frame.builtins();

            assert_eq!(
                builtins
                    .get_item("__name__")
                    .unwrap()
                    .extract::<&str>()
                    .unwrap(),
                "builtins"
            );
            assert!(builtins.contains("len").unwrap());
        })
    }

    #[test]
    #[cfg(all(Py_3_11, not(Py_LIMITED_API)))]
    #[should_panic(expected = "`PyFrame_GetBuiltins` returns a `dict`")]
    fn test_frame_builtins_panics_on_type_error() {
        use crate::types::PyAnyMethods as _;
        Python::attach(|py| -> PyResult<()> {
            let sys = py.import("sys")?;
            let globals = PyDict::new(py);
            globals.set_item("getframe", sys.getattr("_getframe")?)?;
            globals.set_item("__builtins__", py.eval(c"object()", None, None)?)?;
            py.run(c"frame = getframe()", Some(&globals), None)?;
            let frame: Bound<'_, PyFrame> = globals.get_item("frame")?.cast_into()?;

            // This should panic, as `__builtins__` is not a dict
            let _builtins = frame.builtins();

            Ok(())
        })
        .unwrap();
    }

    #[test]
    #[cfg(all(Py_3_11, not(Py_LIMITED_API)))]
    fn test_frame_get_globals() {
        Python::attach(|py| {
            use crate::types::PyAnyMethods as _;

            let frame = get_frame(py);
            let globals = frame.globals();

            assert_eq!(
                globals
                    .get_item("CONST")
                    .unwrap()
                    .extract::<&str>()
                    .unwrap(),
                "global"
            );
        })
    }

    #[test]
    #[cfg(all(Py_3_11, not(Py_LIMITED_API)))]
    fn test_frame_get_locals() {
        Python::attach(|py| {
            use crate::types::PyAnyMethods as _;

            let frame = get_frame(py);
            let locals = frame.locals();

            assert_eq!(
                locals.get_item("var").unwrap().extract::<u32>().unwrap(),
                42
            );
        })
    }
}