use pyo3::prelude::*;
use serde::{Deserialize, Deserializer, Serialize, Serializer};
#[derive(Clone, PartialEq, Eq, Ord, PartialOrd, Hash)]
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(self.0.clone()).unwrap();
write!(f, "{}", s)
}
}
impl RevisionId {
pub fn as_bytes(&self) -> &[u8] {
&self.0
}
pub fn is_null(&self) -> bool {
self.0 == NULL_REVISION
}
pub fn is_reserved(&self) -> bool {
self.0.starts_with(b":")
}
pub fn null() -> Self {
Self(NULL_REVISION.to_vec())
}
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(self.0.clone()).unwrap().as_str())
}
}
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 FromPyObject<'_> for RevisionId {
fn extract_bound(ob: &Bound<PyAny>) -> PyResult<Self> {
let bytes = ob.extract::<Vec<u8>>()?;
Ok(Self(bytes))
}
}
impl ToPyObject for RevisionId {
fn to_object(&self, py: Python) -> PyObject {
pyo3::types::PyBytes::new_bound(py, &self.0).to_object(py)
}
}
impl IntoPy<PyObject> for RevisionId {
fn into_py(self, py: Python) -> PyObject {
pyo3::types::PyBytes::new_bound(py, self.0.as_slice()).to_object(py)
}
}
impl std::fmt::Display for RevisionId {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
let s = String::from_utf8(self.0.clone()).unwrap();
write!(f, "{}", s)
}
}
pub const CURRENT_REVISION: &[u8] = b"current:";
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");
}
}