breezyshim 0.7.15

Rust shim around the Breezy Python API
Documentation
//! Revision ID type and related functions.
use pyo3::prelude::*;
use serde::{Deserialize, Deserializer, Serialize, Serializer};

#[derive(Clone, PartialEq, Eq, Ord, PartialOrd, Hash)]
/// Represents a unique identifier for a revision in a version control system.
///
/// RevisionId is typically a string in UTF-8 encoding, but is stored as bytes
/// to efficiently handle all possible revision formats across different VCS systems.
pub struct RevisionId(Vec<u8>);

impl std::fmt::Debug for RevisionId {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        let s = String::from_utf8_lossy(&self.0);
        write!(f, "{}", s)
    }
}

impl RevisionId {
    /// Get the raw bytes of the revision ID.
    ///
    /// # Returns
    ///
    /// A slice of the underlying bytes
    pub fn as_bytes(&self) -> &[u8] {
        &self.0
    }

    /// Check if this revision ID is the null revision.
    ///
    /// # Returns
    ///
    /// `true` if this is the null revision, `false` otherwise
    pub fn is_null(&self) -> bool {
        self.0 == NULL_REVISION
    }

    /// Check if this revision ID is a reserved revision.
    ///
    /// Reserved revision IDs start with a colon character.
    ///
    /// # Returns
    ///
    /// `true` if this is a reserved revision, `false` otherwise
    pub fn is_reserved(&self) -> bool {
        self.0.starts_with(b":")
    }

    /// Create a new null revision ID.
    ///
    /// # Returns
    ///
    /// A new RevisionId representing the null revision
    pub fn null() -> Self {
        Self(NULL_REVISION.to_vec())
    }

    /// Get the revision ID as a UTF-8 string.
    ///
    /// # Returns
    ///
    /// The revision ID as a string slice
    ///
    /// # Panics
    ///
    /// Panics if the revision ID is not valid UTF-8
    pub fn as_str(&self) -> &str {
        std::str::from_utf8(&self.0).unwrap()
    }
}

#[cfg(feature = "sqlx")]
use sqlx::{postgres::PgTypeInfo, Postgres};

#[cfg(feature = "sqlx")]
impl sqlx::Type<Postgres> for RevisionId {
    fn type_info() -> PgTypeInfo {
        <String as sqlx::Type<sqlx::Postgres>>::type_info()
    }
}

#[cfg(feature = "sqlx")]
impl sqlx::Encode<'_, Postgres> for RevisionId {
    fn encode_by_ref(
        &self,
        buf: &mut sqlx::postgres::PgArgumentBuffer,
    ) -> Result<sqlx::encode::IsNull, Box<dyn std::error::Error + Send + Sync>> {
        sqlx::Encode::<Postgres>::encode_by_ref(&self.as_str(), buf)
    }
}

#[cfg(feature = "sqlx")]
impl sqlx::Decode<'_, Postgres> for RevisionId {
    fn decode(
        value: sqlx::postgres::PgValueRef<'_>,
    ) -> Result<Self, Box<dyn std::error::Error + Send + Sync>> {
        let s: &str = sqlx::Decode::<Postgres>::decode(value)?;
        Ok(RevisionId::from(s.as_bytes()))
    }
}

impl From<Vec<u8>> for RevisionId {
    fn from(value: Vec<u8>) -> Self {
        Self(value)
    }
}

impl From<&[u8]> for RevisionId {
    fn from(value: &[u8]) -> Self {
        Self(value.to_vec())
    }
}

impl Serialize for RevisionId {
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where
        S: Serializer,
    {
        serializer.serialize_str(&String::from_utf8_lossy(&self.0))
    }
}

impl<'de> Deserialize<'de> for RevisionId {
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
    where
        D: Deserializer<'de>,
    {
        String::deserialize(deserializer).map(|s| Self(s.into_bytes()))
    }
}

impl<'a, 'py> FromPyObject<'a, 'py> for RevisionId {
    type Error = PyErr;

    fn extract(ob: Borrowed<'a, 'py, PyAny>) -> PyResult<Self> {
        let bytes = ob.extract::<Vec<u8>>()?;
        Ok(Self(bytes))
    }
}

impl<'py> IntoPyObject<'py> for RevisionId {
    type Target = PyAny;
    type Output = Bound<'py, Self::Target>;
    type Error = std::convert::Infallible;

    fn into_pyobject(self, py: Python<'py>) -> Result<Self::Output, Self::Error> {
        Ok(pyo3::types::PyBytes::new(py, &self.0).into_any())
    }
}

// IntoPy is replaced by IntoPyObject in PyO3 0.25
// The IntoPyObject implementation above handles the conversion

impl std::fmt::Display for RevisionId {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        let s = String::from_utf8_lossy(&self.0);
        write!(f, "{}", s)
    }
}

/// Constant representing the "current" revision identifier.
///
/// This is used to refer to the current revision in working trees.
pub const CURRENT_REVISION: &[u8] = b"current:";

/// Constant representing the "null" revision identifier.
///
/// The null revision is used to represent the absence of a revision,
/// such as the parent of the first commit in a repository.
pub const NULL_REVISION: &[u8] = b"null:";

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

    #[test]
    fn test_revision_id() {
        let id = RevisionId::null();
        assert!(id.is_null());
        assert!(!id.is_reserved());
    }

    #[test]
    fn test_revision_id_from_vec() {
        let id = RevisionId::from(b"test".to_vec());
        assert!(!id.is_null());
        assert!(!id.is_reserved());
    }

    #[test]
    fn test_reserved_revision_id() {
        let id = RevisionId::from(b":test".to_vec());
        assert!(!id.is_null());
        assert!(id.is_reserved());
    }

    #[test]
    fn test_as_bytes() {
        let id = RevisionId::from(b"test".to_vec());
        assert_eq!(id.as_bytes(), b"test");
    }
}