revision 0.20.0

A serialization and deserialization implementation which allows for schema-evolution.
Documentation
use std::borrow::Cow;

use crate::{DeserializeRevisioned, Revisioned, SerializeRevisioned};

impl<T> SerializeRevisioned for Cow<'_, T>
where
	T: Sized + ToOwned + SerializeRevisioned,
	T::Owned: SerializeRevisioned,
{
	#[inline]
	fn serialize_revisioned<W: std::io::Write>(&self, w: &mut W) -> Result<(), crate::Error> {
		match self {
			Cow::Borrowed(b) => b.serialize_revisioned(w),
			Cow::Owned(o) => o.serialize_revisioned(w),
		}
	}
}

impl<T> DeserializeRevisioned for Cow<'_, T>
where
	T: Sized + ToOwned + DeserializeRevisioned,
	T::Owned: DeserializeRevisioned,
{
	#[inline]
	fn deserialize_revisioned<R: std::io::Read>(r: &mut R) -> Result<Self, crate::Error> {
		T::Owned::deserialize_revisioned(r).map(Cow::Owned)
	}
}

impl<T> Revisioned for Cow<'_, T>
where
	T: Sized + ToOwned + Revisioned,
	T::Owned: Revisioned,
{
	#[inline]
	fn revision() -> u16 {
		T::revision()
	}
}

// Specialized implementations for Cow<'_, str>
impl SerializeRevisioned for Cow<'_, str> {
	#[inline]
	fn serialize_revisioned<W: std::io::Write>(&self, w: &mut W) -> Result<(), crate::Error> {
		match self {
			Cow::Borrowed(s) => s.serialize_revisioned(w),
			Cow::Owned(s) => s.serialize_revisioned(w),
		}
	}
}

impl DeserializeRevisioned for Cow<'_, str> {
	#[inline]
	fn deserialize_revisioned<R: std::io::Read>(r: &mut R) -> Result<Self, crate::Error> {
		String::deserialize_revisioned(r).map(Cow::Owned)
	}
}

impl Revisioned for Cow<'_, str> {
	#[inline]
	fn revision() -> u16 {
		1
	}
}

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

	#[test]
	fn cow_borrow() {
		let number = 20u8;

		let cow = Cow::Borrowed(&number);
		let mut mem = Vec::new();
		cow.serialize_revisioned(&mut mem).unwrap();
		assert_eq!(mem.len(), 1);
		let out = Cow::<u8>::deserialize_revisioned(&mut mem.as_slice()).unwrap();
		assert!(matches!(out, Cow::Owned(_)));
		assert_eq!(*out, number)
	}

	#[test]
	fn cow_owned() {
		let number = 20u8;

		let cow: Cow<u8> = Cow::Owned(number);
		let mut mem = Vec::new();
		cow.serialize_revisioned(&mut mem).unwrap();
		assert_eq!(mem.len(), 1);
		let out = Cow::<u8>::deserialize_revisioned(&mut mem.as_slice()).unwrap();
		assert!(matches!(out, Cow::Owned(_)));
		assert_eq!(*out, number)
	}

	#[test]
	fn cow_static_str() {
		let text: &'static str = "hello world";
		let cow: Cow<'static, str> = Cow::Borrowed(text);

		let mut mem = Vec::new();
		cow.serialize_revisioned(&mut mem).unwrap();
		#[cfg(not(feature = "fixed-width-encoding"))]
		assert_eq!(mem.len(), 12); // 11 chars + 1 byte for length encoding
		#[cfg(feature = "fixed-width-encoding")]
		assert_eq!(mem.len(), 19); // 11 chars + 8 bytes for length encoding

		let out = Cow::<'static, str>::deserialize_revisioned(&mut mem.as_slice()).unwrap();
		assert!(matches!(out, Cow::Owned(_)));
		assert_eq!(&*out, text);
	}

	#[test]
	fn cow_owned_string_as_static_str() {
		let owned_text = "hello world".to_string();
		let cow: Cow<'static, str> = Cow::Owned(owned_text.clone());

		let mut mem = Vec::new();
		cow.serialize_revisioned(&mut mem).unwrap();
		#[cfg(not(feature = "fixed-width-encoding"))]
		assert_eq!(mem.len(), 12); // 11 chars + 1 byte for length encoding
		#[cfg(feature = "fixed-width-encoding")]
		assert_eq!(mem.len(), 19); // 11 chars + 8 bytes for length encoding

		let out = Cow::<'static, str>::deserialize_revisioned(&mut mem.as_slice()).unwrap();
		assert!(matches!(out, Cow::Owned(_)));
		assert_eq!(&*out, &owned_text);
	}
}