revision 0.20.0

A serialization and deserialization implementation which allows for schema-evolution.
Documentation
use super::super::Error;
use super::super::{DeserializeRevisioned, Revisioned, SerializeRevisioned};

macro_rules! impl_tuple {
	($($n:ident),*$(,)?) => {
		impl_tuple!{@marker $($n,)*}
	};

	($($n:ident,)* @marker $head:ident, $($tail:ident,)*) => {
		impl<$($n),*> SerializeRevisioned for ($($n,)*)
			where $($n: SerializeRevisioned),*
		{
			#[inline]
			#[allow(non_snake_case)]
			fn serialize_revisioned<W: std::io::Write>(&self, _writer: &mut W) -> Result<(), Error> {
				let ($(ref $n,)*) = *self;
				$(
					$n.serialize_revisioned(_writer)?;
				)*
				Ok(())
			}
		}

		impl<$($n),*> DeserializeRevisioned for ($($n,)*)
			where $($n: DeserializeRevisioned),*
		{
			#[inline]
			#[allow(non_snake_case)]
			fn deserialize_revisioned<R: std::io::Read>(_reader: &mut R) -> Result<Self, Error> {
				$(
					let $n = DeserializeRevisioned::deserialize_revisioned(_reader)?;
				)*
				Ok(($($n,)*))
			}
		}

		impl<$($n),*> Revisioned for ($($n,)*)
			where $($n: Revisioned),*
		{
			#[inline]
			fn revision() -> u16{
				1
			}
		}

		impl_tuple!{$($n,)* $head, @marker $($tail,)*}

	};
	($($n:ident,)* @marker) => {
		impl<$($n),*> SerializeRevisioned for ($($n),*)
			where $($n: SerializeRevisioned),*
		{
			#[inline]
			#[allow(non_snake_case)]
			fn serialize_revisioned<W: std::io::Write>(&self, _writer: &mut W) -> Result<(), Error> {
				let ($($n),*) = self;
				$(
					$n.serialize_revisioned(_writer)?;
				)*
				Ok(())
			}
		}

		impl<$($n),*> DeserializeRevisioned for ($($n),*)
			where $($n: DeserializeRevisioned),*
		{
			#[inline]
			#[allow(non_snake_case)]
			fn deserialize_revisioned<R: std::io::Read>(_reader: &mut R) -> Result<Self, Error> {
				$(
					let $n = DeserializeRevisioned::deserialize_revisioned(_reader)?;
				)*
				Ok(($($n),*))
			}
		}

		impl<$($n),*> Revisioned for ($($n),*)
			where $($n: Revisioned),*
		{
			#[inline]
			fn revision() -> u16{
				1
			}
		}
	};
}

impl_tuple! { A,B,C,D,E,F }

#[cfg(test)]
mod tests {

	use crate::implementations::assert_bincode_compat;

	use super::*;

	#[test]
	fn test_tuple_2() {
		let val = (String::from("test"), true);
		assert_bincode_compat(&val);
		let mut mem: Vec<u8> = vec![];
		val.serialize_revisioned(&mut mem).unwrap();
		#[cfg(not(feature = "fixed-width-encoding"))]
		assert_eq!(mem.len(), 6);
		#[cfg(feature = "fixed-width-encoding")]
		assert_eq!(mem.len(), 13);
		let out =
			<(String, bool) as DeserializeRevisioned>::deserialize_revisioned(&mut mem.as_slice())
				.unwrap();
		assert_eq!(val, out);
	}

	#[test]
	fn test_tuple_3() {
		let val = (String::from("test"), true, 1419247293847192847.13947134978139487);
		assert_bincode_compat(&val);
		let mut mem: Vec<u8> = vec![];
		val.serialize_revisioned(&mut mem).unwrap();
		#[cfg(not(feature = "fixed-width-encoding"))]
		assert_eq!(mem.len(), 14);
		#[cfg(feature = "fixed-width-encoding")]
		assert_eq!(mem.len(), 21);
		let out = <(String, bool, f64) as DeserializeRevisioned>::deserialize_revisioned(
			&mut mem.as_slice(),
		)
		.unwrap();
		assert_eq!(val, out);
	}

	#[test]
	fn test_tuple_4() {
		let val = (String::from("test"), true, 1419247293847192847.13947134978139487, Some('t'));
		assert_bincode_compat(&val);
		let mut mem: Vec<u8> = vec![];
		val.serialize_revisioned(&mut mem).unwrap();
		#[cfg(not(feature = "fixed-width-encoding"))]
		assert_eq!(mem.len(), 16);
		#[cfg(feature = "fixed-width-encoding")]
		assert_eq!(mem.len(), 23);
		let out =
			<(String, bool, f64, Option<char>) as DeserializeRevisioned>::deserialize_revisioned(
				&mut mem.as_slice(),
			)
			.unwrap();
		assert_eq!(val, out);
	}

	#[test]
	fn test_tuple_5() {
		let val = (
			String::from("test"),
			true,
			1419247293847192847.13947134978139487,
			Some('t'),
			vec![4u8, 19u8, 133u8],
		);
		assert_bincode_compat(&val);
		let mut mem: Vec<u8> = vec![];
		val.serialize_revisioned(&mut mem).unwrap();
		#[cfg(not(feature = "fixed-width-encoding"))]
		assert_eq!(mem.len(), 20);
		#[cfg(feature = "fixed-width-encoding")]
		assert_eq!(mem.len(), 34);
		let out =
			<(String, bool, f64, Option<char>, Vec<u8>) as DeserializeRevisioned>::deserialize_revisioned(
				&mut mem.as_slice(),
			)
			.unwrap();
		assert_eq!(val, out);
	}
}