216 lines
5.0 KiB
Rust
216 lines
5.0 KiB
Rust
|
use ssz::{Decode, Encode};
|
||
|
use ssz_derive::{Decode, Encode};
|
||
|
use std::fmt::Debug;
|
||
|
use std::marker::PhantomData;
|
||
|
|
||
|
fn assert_encode<T: Encode>(item: &T, bytes: &[u8]) {
|
||
|
assert_eq!(item.as_ssz_bytes(), bytes);
|
||
|
}
|
||
|
|
||
|
fn assert_encode_decode<T: Encode + Decode + PartialEq + Debug>(item: &T, bytes: &[u8]) {
|
||
|
assert_encode(item, bytes);
|
||
|
assert_eq!(T::from_ssz_bytes(bytes).unwrap(), *item);
|
||
|
}
|
||
|
|
||
|
#[derive(PartialEq, Debug, Encode, Decode)]
|
||
|
#[ssz(enum_behaviour = "union")]
|
||
|
enum TwoFixedUnion {
|
||
|
U8(u8),
|
||
|
U16(u16),
|
||
|
}
|
||
|
|
||
|
#[derive(PartialEq, Debug, Encode, Decode)]
|
||
|
struct TwoFixedUnionStruct {
|
||
|
a: TwoFixedUnion,
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn two_fixed_union() {
|
||
|
let eight = TwoFixedUnion::U8(1);
|
||
|
let sixteen = TwoFixedUnion::U16(1);
|
||
|
|
||
|
assert_encode_decode(&eight, &[0, 1]);
|
||
|
assert_encode_decode(&sixteen, &[1, 1, 0]);
|
||
|
|
||
|
assert_encode_decode(&TwoFixedUnionStruct { a: eight }, &[4, 0, 0, 0, 0, 1]);
|
||
|
assert_encode_decode(&TwoFixedUnionStruct { a: sixteen }, &[4, 0, 0, 0, 1, 1, 0]);
|
||
|
}
|
||
|
|
||
|
#[derive(PartialEq, Debug, Encode, Decode)]
|
||
|
struct VariableA {
|
||
|
a: u8,
|
||
|
b: Vec<u8>,
|
||
|
}
|
||
|
|
||
|
#[derive(PartialEq, Debug, Encode, Decode)]
|
||
|
struct VariableB {
|
||
|
a: Vec<u8>,
|
||
|
b: u8,
|
||
|
}
|
||
|
|
||
|
#[derive(PartialEq, Debug, Encode)]
|
||
|
#[ssz(enum_behaviour = "transparent")]
|
||
|
enum TwoVariableTrans {
|
||
|
A(VariableA),
|
||
|
B(VariableB),
|
||
|
}
|
||
|
|
||
|
#[derive(PartialEq, Debug, Encode)]
|
||
|
struct TwoVariableTransStruct {
|
||
|
a: TwoVariableTrans,
|
||
|
}
|
||
|
|
||
|
#[derive(PartialEq, Debug, Encode, Decode)]
|
||
|
#[ssz(enum_behaviour = "union")]
|
||
|
enum TwoVariableUnion {
|
||
|
A(VariableA),
|
||
|
B(VariableB),
|
||
|
}
|
||
|
|
||
|
#[derive(PartialEq, Debug, Encode, Decode)]
|
||
|
struct TwoVariableUnionStruct {
|
||
|
a: TwoVariableUnion,
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn two_variable_trans() {
|
||
|
let trans_a = TwoVariableTrans::A(VariableA {
|
||
|
a: 1,
|
||
|
b: vec![2, 3],
|
||
|
});
|
||
|
let trans_b = TwoVariableTrans::B(VariableB {
|
||
|
a: vec![1, 2],
|
||
|
b: 3,
|
||
|
});
|
||
|
|
||
|
assert_encode(&trans_a, &[1, 5, 0, 0, 0, 2, 3]);
|
||
|
assert_encode(&trans_b, &[5, 0, 0, 0, 3, 1, 2]);
|
||
|
|
||
|
assert_encode(
|
||
|
&TwoVariableTransStruct { a: trans_a },
|
||
|
&[4, 0, 0, 0, 1, 5, 0, 0, 0, 2, 3],
|
||
|
);
|
||
|
assert_encode(
|
||
|
&TwoVariableTransStruct { a: trans_b },
|
||
|
&[4, 0, 0, 0, 5, 0, 0, 0, 3, 1, 2],
|
||
|
);
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn two_variable_union() {
|
||
|
let union_a = TwoVariableUnion::A(VariableA {
|
||
|
a: 1,
|
||
|
b: vec![2, 3],
|
||
|
});
|
||
|
let union_b = TwoVariableUnion::B(VariableB {
|
||
|
a: vec![1, 2],
|
||
|
b: 3,
|
||
|
});
|
||
|
|
||
|
assert_encode_decode(&union_a, &[0, 1, 5, 0, 0, 0, 2, 3]);
|
||
|
assert_encode_decode(&union_b, &[1, 5, 0, 0, 0, 3, 1, 2]);
|
||
|
|
||
|
assert_encode_decode(
|
||
|
&TwoVariableUnionStruct { a: union_a },
|
||
|
&[4, 0, 0, 0, 0, 1, 5, 0, 0, 0, 2, 3],
|
||
|
);
|
||
|
assert_encode_decode(
|
||
|
&TwoVariableUnionStruct { a: union_b },
|
||
|
&[4, 0, 0, 0, 1, 5, 0, 0, 0, 3, 1, 2],
|
||
|
);
|
||
|
}
|
||
|
|
||
|
#[derive(PartialEq, Debug, Encode, Decode)]
|
||
|
#[ssz(enum_behaviour = "union")]
|
||
|
enum TwoVecUnion {
|
||
|
A(Vec<u8>),
|
||
|
B(Vec<u8>),
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn two_vec_union() {
|
||
|
assert_encode_decode(&TwoVecUnion::A(vec![]), &[0]);
|
||
|
assert_encode_decode(&TwoVecUnion::B(vec![]), &[1]);
|
||
|
|
||
|
assert_encode_decode(&TwoVecUnion::A(vec![0]), &[0, 0]);
|
||
|
assert_encode_decode(&TwoVecUnion::B(vec![0]), &[1, 0]);
|
||
|
|
||
|
assert_encode_decode(&TwoVecUnion::A(vec![0, 1]), &[0, 0, 1]);
|
||
|
assert_encode_decode(&TwoVecUnion::B(vec![0, 1]), &[1, 0, 1]);
|
||
|
}
|
||
|
|
||
|
#[derive(PartialEq, Debug, Encode, Decode)]
|
||
|
#[ssz(struct_behaviour = "transparent")]
|
||
|
struct TransparentStruct {
|
||
|
inner: Vec<u8>,
|
||
|
}
|
||
|
|
||
|
impl TransparentStruct {
|
||
|
fn new(inner: u8) -> Self {
|
||
|
Self { inner: vec![inner] }
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn transparent_struct() {
|
||
|
assert_encode_decode(&TransparentStruct::new(42), &vec![42_u8].as_ssz_bytes());
|
||
|
}
|
||
|
|
||
|
#[derive(PartialEq, Debug, Encode, Decode)]
|
||
|
#[ssz(struct_behaviour = "transparent")]
|
||
|
struct TransparentStructSkippedField {
|
||
|
inner: Vec<u8>,
|
||
|
#[ssz(skip_serializing, skip_deserializing)]
|
||
|
skipped: PhantomData<u64>,
|
||
|
}
|
||
|
|
||
|
impl TransparentStructSkippedField {
|
||
|
fn new(inner: u8) -> Self {
|
||
|
Self {
|
||
|
inner: vec![inner],
|
||
|
skipped: PhantomData,
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn transparent_struct_skipped_field() {
|
||
|
assert_encode_decode(
|
||
|
&TransparentStructSkippedField::new(42),
|
||
|
&vec![42_u8].as_ssz_bytes(),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
#[derive(PartialEq, Debug, Encode, Decode)]
|
||
|
#[ssz(struct_behaviour = "transparent")]
|
||
|
struct TransparentStructNewType(Vec<u8>);
|
||
|
|
||
|
#[test]
|
||
|
fn transparent_struct_newtype() {
|
||
|
assert_encode_decode(
|
||
|
&TransparentStructNewType(vec![42_u8]),
|
||
|
&vec![42_u8].as_ssz_bytes(),
|
||
|
);
|
||
|
}
|
||
|
|
||
|
#[derive(PartialEq, Debug, Encode, Decode)]
|
||
|
#[ssz(struct_behaviour = "transparent")]
|
||
|
struct TransparentStructNewTypeSkippedField(
|
||
|
Vec<u8>,
|
||
|
#[ssz(skip_serializing, skip_deserializing)] PhantomData<u64>,
|
||
|
);
|
||
|
|
||
|
impl TransparentStructNewTypeSkippedField {
|
||
|
fn new(inner: Vec<u8>) -> Self {
|
||
|
Self(inner, PhantomData)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#[test]
|
||
|
fn transparent_struct_newtype_skipped_field() {
|
||
|
assert_encode_decode(
|
||
|
&TransparentStructNewTypeSkippedField::new(vec![42_u8]),
|
||
|
&vec![42_u8].as_ssz_bytes(),
|
||
|
);
|
||
|
}
|