libcst 1.8.6

A Python parser and Concrete Syntax Tree library.
Documentation
// Copyright (c) Meta Platforms, Inc. and affiliates.
//
// This source code is licensed under the MIT license found in the
// LICENSE file in the root directory of this source tree.

use pyo3::prelude::*;
use std::convert::AsRef;
use std::ops::Deref;

/// An immutable wrapper around a rust type T and its Py<PyAny> equivalent. Caches the conversion
/// to and from the Py<PyAny>.
pub struct PyCached<T> {
    native: T,
    py_object: Py<PyAny>,
}

impl<T> PyCached<T>
where
    T: ToPyObject,
{
    pub fn new(py: Python, native: T) -> Self {
        Self {
            py_object: native.to_object(py),
            native,
        }
    }
}

impl<'source, T> FromPyObject<'source> for PyCached<T>
where
    T: FromPyObject<'source>,
{
    fn extract(ob: &'source PyAny) -> PyResult<Self> {
        Python::attach(|py| {
            Ok(PyCached {
                native: ob.extract()?,
                py_object: ob.to_object(py),
            })
        })
    }
}

impl<T> IntoPy<Py<PyAny>> for PyCached<T> {
    fn into_py(self, _py: Python) -> Py<PyAny> {
        self.py_object
    }
}

impl<T> ToPyObject for PyCached<T> {
    fn to_object(&self, py: Python) -> Py<PyAny> {
        self.py_object.clone_ref(py)
    }
}

impl<T> AsRef<T> for PyCached<T> {
    fn as_ref(&self) -> &T {
        &self.native
    }
}

impl<T> Deref for PyCached<T> {
    type Target = T;

    fn deref(&self) -> &Self::Target {
        &self.native
    }
}

impl<T> From<T> for PyCached<T>
where
    T: ToPyObject,
{
    fn from(val: T) -> Self {
        Python::attach(|py| Self::new(py, val))
    }
}