use serde::{ser, Serialize};
use thiserror::Error;
#[derive(Debug, Error)]
pub enum RlpError {
#[error("Unknown error,{0}")]
Unknown(String),
#[error("Serilaize multi-element without calling begin_list")]
List,
#[error("Call finalize without calling end_list")]
UnclosedList,
#[error("call end_list before calling begin_list")]
UnopenList,
#[error("Unsupport serialize type, {0}")]
UnsupportType(String),
}
impl ser::Error for RlpError {
fn custom<T>(msg: T) -> Self
where
T: std::fmt::Display,
{
Self::Unknown(msg.to_string())
}
}
#[derive(Debug, Default)]
struct RlpList {
buff: Vec<u8>,
}
impl RlpList {
fn finalize(mut self) -> Result<Vec<u8>, RlpError> {
let len = self.buff.len();
if len <= 55 {
let mut buff = vec![0xc0u8 + len as u8];
buff.append(&mut self.buff);
Ok(buff)
} else {
let mut len_buff = unsigned_to_buff(&len.to_be_bytes()).to_vec();
let mut buff = vec![0xf7 + len_buff.len() as u8];
buff.append(&mut len_buff);
buff.append(&mut self.buff);
Ok(buff)
}
}
pub fn rlp_append_string(&mut self, bytes: &[u8]) -> Result<(), RlpError> {
self.buff.append(&mut bytes.to_owned());
Ok(())
}
}
fn rlp_encode_string(bytes: &[u8]) -> Result<Vec<u8>, RlpError> {
let len = bytes.len();
match len {
0 => Ok(vec![0x80u8]),
len @ 1..=55 => {
if len == 1 && bytes[0] < 0x80 {
Ok(bytes.to_owned())
} else {
let mut buff = vec![0x80u8 + len as u8];
buff.append(&mut bytes.to_owned());
Ok(buff)
}
}
len => {
let mut len_buff = unsigned_to_buff(&len.to_be_bytes()).to_vec();
let mut buff = vec![0xb7 + len_buff.len() as u8];
buff.append(&mut len_buff);
buff.append(&mut bytes.to_owned());
Ok(buff)
}
}
}
#[derive(Debug, Default)]
pub struct RlpEncoder {
buff: Vec<u8>,
list_stack: Vec<RlpList>,
}
impl RlpEncoder {
pub fn begin_list(&mut self) -> Result<(), RlpError> {
self.list_stack.push(Default::default());
Ok(())
}
pub fn append_string(&mut self, bytes: &[u8]) -> Result<(), RlpError> {
if !self.list_stack.is_empty() {
self.list_stack
.last_mut()
.unwrap()
.rlp_append_string(&rlp_encode_string(bytes)?)?;
} else {
if self.buff.len() != 0 {
return Err(RlpError::List.into());
} else {
self.buff = rlp_encode_string(bytes)?;
}
}
Ok(())
}
pub fn end_list(&mut self) -> Result<(), RlpError> {
if let Some(list) = self.list_stack.pop() {
let mut buff = list.finalize()?;
if !self.list_stack.is_empty() {
self.list_stack
.last_mut()
.unwrap()
.rlp_append_string(&buff)?;
} else {
self.buff.append(&mut buff);
}
return Ok(());
} else {
return Err(RlpError::UnopenList.into());
}
}
pub fn finalize(self) -> Result<Vec<u8>, RlpError> {
if !self.list_stack.is_empty() {
return Err(RlpError::UnclosedList.into());
}
Ok(self.buff)
}
}
fn signed_to_buff(bytes: &[u8]) -> &[u8] {
let lead_ones = bytes.iter().take_while(|c| **c == 0xff).count();
let lead_zeros = bytes.iter().take_while(|c| **c == 0x00).count();
if lead_ones > 0 {
&bytes[(lead_ones - 1)..]
} else {
&bytes[lead_zeros..]
}
}
fn unsigned_to_buff(bytes: &[u8]) -> &[u8] {
let lead_zeros = bytes.iter().take_while(|c| **c == 0x00).count();
&bytes[lead_zeros..]
}
impl<'a> ser::Serializer for &'a mut RlpEncoder {
type Ok = ();
type Error = RlpError;
type SerializeSeq = Self;
type SerializeTuple = Self;
type SerializeTupleStruct = Self;
type SerializeTupleVariant = Self;
type SerializeMap = Self;
type SerializeStruct = Self;
type SerializeStructVariant = Self;
fn is_human_readable(&self) -> bool {
false
}
fn serialize_bool(self, _v: bool) -> Result<Self::Ok, Self::Error> {
Err(RlpError::UnsupportType("bool".to_owned()))
}
fn serialize_bytes(self, v: &[u8]) -> Result<Self::Ok, Self::Error> {
self.append_string(v)
}
fn serialize_char(self, _v: char) -> Result<Self::Ok, Self::Error> {
unimplemented!("Contract abi don't support rust char")
}
fn serialize_f32(self, v: f32) -> Result<Self::Ok, Self::Error> {
self.serialize_f64(v as f64)
}
fn serialize_f64(self, _v: f64) -> Result<Self::Ok, Self::Error> {
Err(RlpError::UnsupportType("f64".to_owned()))
}
fn serialize_i128(self, v: i128) -> Result<Self::Ok, Self::Error> {
let bytes = v.to_be_bytes();
self.append_string(signed_to_buff(&bytes))
}
fn serialize_i16(self, v: i16) -> Result<Self::Ok, Self::Error> {
let bytes = v.to_be_bytes();
self.append_string(signed_to_buff(&bytes))
}
fn serialize_i32(self, v: i32) -> Result<Self::Ok, Self::Error> {
let bytes = v.to_be_bytes();
self.append_string(signed_to_buff(&bytes))
}
fn serialize_i64(self, v: i64) -> Result<Self::Ok, Self::Error> {
let bytes = v.to_be_bytes();
self.append_string(signed_to_buff(&bytes))
}
fn serialize_i8(self, v: i8) -> Result<Self::Ok, Self::Error> {
let bytes = v.to_be_bytes();
self.append_string(signed_to_buff(&bytes))
}
fn serialize_map(self, _len: Option<usize>) -> Result<Self::SerializeMap, Self::Error> {
Err(RlpError::UnsupportType("map".to_owned()))
}
fn serialize_newtype_struct<T: ?Sized>(
self,
name: &'static str,
value: &T,
) -> Result<Self::Ok, Self::Error>
where
T: serde::Serialize,
{
match name {
"bytes" | "bytesN" | "address" | "uint256" | "int256" => {
let bytes = unsafe { (value as *const T).cast::<Vec<u8>>().as_ref().unwrap() };
self.append_string(bytes)
}
_ => {
self.begin_list()?;
value.serialize(&mut *self)?;
self.end_list()?;
Ok(())
}
}
}
fn serialize_newtype_variant<T: ?Sized>(
self,
_name: &'static str,
_variant_index: u32,
_variant: &'static str,
_value: &T,
) -> Result<Self::Ok, Self::Error>
where
T: serde::Serialize,
{
unimplemented!("Contract abi don't support rust enum")
}
fn serialize_none(self) -> Result<Self::Ok, Self::Error> {
self.append_string(&[])
}
fn serialize_seq(self, _len: Option<usize>) -> Result<Self::SerializeSeq, Self::Error> {
self.begin_list()?;
Ok(self)
}
fn serialize_some<T: ?Sized>(self, value: &T) -> Result<Self::Ok, Self::Error>
where
T: serde::Serialize,
{
value.serialize(self)
}
fn serialize_str(self, v: &str) -> Result<Self::Ok, Self::Error> {
self.append_string(v.as_bytes())
}
fn serialize_struct(
self,
_name: &'static str,
_len: usize,
) -> Result<Self::SerializeStruct, Self::Error> {
self.begin_list()?;
Ok(self)
}
fn serialize_struct_variant(
self,
_name: &'static str,
_variant_index: u32,
_variant: &'static str,
_len: usize,
) -> Result<Self::SerializeStructVariant, Self::Error> {
unimplemented!("Contract abi don't support rust enum")
}
fn serialize_tuple(self, _len: usize) -> Result<Self::SerializeTuple, Self::Error> {
self.begin_list()?;
Ok(self)
}
fn serialize_tuple_struct(
self,
_name: &'static str,
_len: usize,
) -> Result<Self::SerializeTupleStruct, Self::Error> {
self.begin_list()?;
Ok(self)
}
fn serialize_tuple_variant(
self,
_name: &'static str,
_variant_index: u32,
_variant: &'static str,
_len: usize,
) -> Result<Self::SerializeTupleVariant, Self::Error> {
unimplemented!("Contract abi don't support rust enum")
}
fn serialize_u128(self, v: u128) -> Result<Self::Ok, Self::Error> {
let bytes = v.to_be_bytes();
let lead_zeros = bytes.iter().take_while(|c| **c == 0x00).count();
self.append_string(&bytes[lead_zeros..])
}
fn serialize_u16(self, v: u16) -> Result<Self::Ok, Self::Error> {
let bytes = v.to_be_bytes();
let lead_zeros = bytes.iter().take_while(|c| **c == 0x00).count();
self.append_string(&bytes[lead_zeros..])
}
fn serialize_u32(self, v: u32) -> Result<Self::Ok, Self::Error> {
let bytes = v.to_be_bytes();
let lead_zeros = bytes.iter().take_while(|c| **c == 0x00).count();
self.append_string(&bytes[lead_zeros..])
}
fn serialize_u64(self, v: u64) -> Result<Self::Ok, Self::Error> {
let bytes = v.to_be_bytes();
let lead_zeros = bytes.iter().take_while(|c| **c == 0x00).count();
self.append_string(&bytes[lead_zeros..])
}
fn serialize_u8(self, v: u8) -> Result<Self::Ok, Self::Error> {
let bytes = v.to_be_bytes();
let lead_zeros = bytes.iter().take_while(|c| **c == 0x00).count();
self.append_string(&bytes[lead_zeros..])
}
fn serialize_unit(self) -> Result<Self::Ok, Self::Error> {
Ok(())
}
fn serialize_unit_struct(self, _: &'static str) -> Result<Self::Ok, Self::Error> {
Ok(())
}
fn serialize_unit_variant(
self,
_name: &'static str,
_variant_index: u32,
_variant: &'static str,
) -> Result<Self::Ok, Self::Error> {
unimplemented!("Contract abi don't support rust enum")
}
}
impl<'a> ser::SerializeMap for &'a mut RlpEncoder {
type Error = RlpError;
type Ok = ();
fn end(self) -> Result<Self::Ok, Self::Error> {
unimplemented!()
}
fn serialize_key<T: ?Sized>(&mut self, _key: &T) -> Result<(), Self::Error>
where
T: serde::Serialize,
{
unimplemented!()
}
fn serialize_value<T: ?Sized>(&mut self, _value: &T) -> Result<(), Self::Error>
where
T: serde::Serialize,
{
unimplemented!()
}
}
impl<'a> ser::SerializeSeq for &'a mut RlpEncoder {
type Error = RlpError;
type Ok = ();
fn end(self) -> Result<Self::Ok, Self::Error> {
self.end_list()?;
Ok(())
}
fn serialize_element<T: ?Sized>(&mut self, value: &T) -> Result<(), Self::Error>
where
T: serde::Serialize,
{
value.serialize(&mut **self)?;
Ok(())
}
}
impl<'a> ser::SerializeStruct for &'a mut RlpEncoder {
type Error = RlpError;
type Ok = ();
fn end(self) -> Result<Self::Ok, Self::Error> {
self.end_list()?;
Ok(())
}
fn serialize_field<T: ?Sized>(
&mut self,
_key: &'static str,
value: &T,
) -> Result<(), Self::Error>
where
T: serde::Serialize,
{
value.serialize(&mut **self)?;
Ok(())
}
}
impl<'a> ser::SerializeStructVariant for &'a mut RlpEncoder {
type Error = RlpError;
type Ok = ();
fn end(self) -> Result<Self::Ok, Self::Error> {
self.end_list()?;
Ok(())
}
fn serialize_field<T: ?Sized>(
&mut self,
_key: &'static str,
value: &T,
) -> Result<(), Self::Error>
where
T: serde::Serialize,
{
value.serialize(&mut **self)?;
Ok(())
}
}
impl<'a> ser::SerializeTuple for &'a mut RlpEncoder {
type Error = RlpError;
type Ok = ();
fn end(self) -> Result<Self::Ok, Self::Error> {
self.end_list()?;
Ok(())
}
fn serialize_element<T: ?Sized>(&mut self, value: &T) -> Result<(), Self::Error>
where
T: serde::Serialize,
{
value.serialize(&mut **self)?;
Ok(())
}
}
impl<'a> ser::SerializeTupleVariant for &'a mut RlpEncoder {
type Error = RlpError;
type Ok = ();
fn end(self) -> Result<Self::Ok, Self::Error> {
self.end_list()?;
Ok(())
}
fn serialize_field<T: ?Sized>(&mut self, value: &T) -> Result<(), Self::Error>
where
T: serde::Serialize,
{
value.serialize(&mut **self)?;
Ok(())
}
}
impl<'a> ser::SerializeTupleStruct for &'a mut RlpEncoder {
type Error = RlpError;
type Ok = ();
fn end(self) -> Result<Self::Ok, Self::Error> {
self.end_list()?;
Ok(())
}
fn serialize_field<T: ?Sized>(&mut self, value: &T) -> Result<(), Self::Error>
where
T: serde::Serialize,
{
value.serialize(&mut **self)?;
Ok(())
}
}
pub fn rlp_encode<S: Serialize + ?Sized>(value: &S) -> Result<Vec<u8>, RlpError> {
let mut serializer = RlpEncoder::default();
value.serialize(&mut serializer)?;
Ok(serializer.finalize()?)
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_rlp_encode() {
assert_eq!(rlp_encode("dog").unwrap(), [0x83u8, b'd', b'o', b'g']);
assert_eq!(
rlp_encode(&vec!["cat".to_string(), "dog".to_string()]).unwrap(),
[0xc8, 0x83u8, b'c', b'a', b't', 0x83, b'd', b'o', b'g']
);
assert_eq!(rlp_encode(&Vec::<String>::new()).unwrap(), [0xc0]);
assert_eq!(rlp_encode(&0usize).unwrap(), [0x80]);
assert_eq!(rlp_encode("\x00").unwrap(), [0x00]);
assert_eq!(rlp_encode("\x04\x00").unwrap(), [0x82, 0x04, 0x00]);
fn test_lists() -> Vec<u8> {
let mut encoder = RlpEncoder::default();
encoder.begin_list().unwrap();
{
encoder.begin_list().unwrap();
encoder.end_list().unwrap();
}
{
encoder.begin_list().unwrap();
{
encoder.begin_list().unwrap();
encoder.end_list().unwrap();
}
encoder.end_list().unwrap();
}
{
encoder.begin_list().unwrap();
{
encoder.begin_list().unwrap();
encoder.end_list().unwrap();
}
{
encoder.begin_list().unwrap();
{
encoder.begin_list().unwrap();
encoder.end_list().unwrap();
}
encoder.end_list().unwrap();
}
encoder.end_list().unwrap();
}
encoder.end_list().unwrap();
encoder.finalize().unwrap()
}
assert_eq!(
test_lists(),
[0xc7, 0xc0, 0xc1, 0xc0, 0xc3, 0xc0, 0xc1, 0xc0]
);
let mut expected = [0xb8, 0x38].to_vec();
expected.append(&mut b"Lorem ipsum dolor sit amet, consectetur adipisicing elit".to_vec());
assert_eq!(
rlp_encode("Lorem ipsum dolor sit amet, consectetur adipisicing elit").unwrap(),
expected
);
}
#[test]
fn test_option() {
assert_eq!([0x80u8].as_slice(), rlp_encode("").unwrap());
}
}