diff --git a/borsh/src/schema.rs b/borsh/src/schema.rs index 7c5587065..0139f7dc0 100644 --- a/borsh/src/schema.rs +++ b/borsh/src/schema.rs @@ -16,7 +16,7 @@ use crate as borsh; // For `#[derive(BorshSerialize, BorshDeserialize)]`. use crate::__private::maybestd::{ borrow, boxed::Box, - collections::{btree_map::Entry, BTreeMap, BTreeSet}, + collections::{btree_map::Entry, BTreeMap, BTreeSet, LinkedList, VecDeque}, format, string::{String, ToString}, vec, @@ -635,25 +635,33 @@ where } } -impl BorshSchema for Vec -where - T: BorshSchema, -{ - fn add_definitions_recursively(definitions: &mut BTreeMap) { - let definition = Definition::Sequence { - length_width: Definition::DEFAULT_LENGTH_WIDTH, - length_range: Definition::DEFAULT_LENGTH_RANGE, - elements: T::declaration(), - }; - add_definition(Self::declaration(), definition, definitions); - T::add_definitions_recursively(definitions); - } +macro_rules! impl_for_vec_like_collection { + ($type: ident) => { + impl BorshSchema for $type + where + T: BorshSchema, + { + fn add_definitions_recursively(definitions: &mut BTreeMap) { + let definition = Definition::Sequence { + length_width: Definition::DEFAULT_LENGTH_WIDTH, + length_range: Definition::DEFAULT_LENGTH_RANGE, + elements: T::declaration(), + }; + add_definition(Self::declaration(), definition, definitions); + T::add_definitions_recursively(definitions); + } - fn declaration() -> Declaration { - format!(r#"Vec<{}>"#, T::declaration()) - } + fn declaration() -> Declaration { + format!(r#"Vec<{}>"#, T::declaration()) + } + } + }; } +impl_for_vec_like_collection!(Vec); +impl_for_vec_like_collection!(VecDeque); +impl_for_vec_like_collection!(LinkedList); + impl BorshSchema for [T] where T: BorshSchema, diff --git a/borsh/tests/schema/test_vecs.rs b/borsh/tests/schema/test_vecs.rs index cbc8f6109..e2acb0086 100644 --- a/borsh/tests/schema/test_vecs.rs +++ b/borsh/tests/schema/test_vecs.rs @@ -1,24 +1,35 @@ use crate::common_macro::schema_imports::*; +use alloc::collections::{VecDeque, LinkedList}; -#[test] -fn simple_vec() { - let actual_name = Vec::::declaration(); - let mut actual_defs = schema_map!(); - Vec::::add_definitions_recursively(&mut actual_defs); - assert_eq!("Vec", actual_name); - assert_eq!( - schema_map! { - "Vec" => Definition::Sequence { - length_width: Definition::DEFAULT_LENGTH_WIDTH, - length_range: Definition::DEFAULT_LENGTH_RANGE, - elements: "u64".to_string(), - }, - "u64" => Definition::Primitive(8) - }, - actual_defs - ); +macro_rules! test_vec_like_collectioan_schema { + [$test_name: ident, $type: ident] => [ + + #[test] + fn $test_name() { + let actual_name = $type::::declaration(); + let mut actual_defs = schema_map!(); + $type::::add_definitions_recursively(&mut actual_defs); + + assert_eq!("Vec", actual_name); + assert_eq!( + schema_map! { + "Vec" => Definition::Sequence { + length_width: Definition::DEFAULT_LENGTH_WIDTH, + length_range: Definition::DEFAULT_LENGTH_RANGE, + elements: "u64".to_string(), + }, + "u64" => Definition::Primitive(8) + }, + actual_defs + ); + } + ]; } +test_vec_like_collectioan_schema!(simple_vec, Vec); +test_vec_like_collectioan_schema!(vec_deque, VecDeque); +test_vec_like_collectioan_schema!(linked_list, LinkedList); + #[test] fn nested_vec() { let actual_name = Vec::>::declaration();