1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
use stellar_xdr::ScObjectType;
use crate::{ConversionError, Env, Object, RawVal, RawValConvertible, TryFromVal, TryIntoVal};
macro_rules! impl_for_tuple {
( $count:literal $count_usize:literal $($typ:ident $idx:tt)+ ) => {
impl<E: Env, $($typ),*> TryFromVal<E, RawVal> for ($($typ,)*)
where
$($typ: TryFromVal<E, RawVal>),*
{
type Error = ConversionError;
fn try_from_val(env: &E, val: &RawVal) -> Result<Self, Self::Error> {
let val = *val;
if !Object::val_is_obj_type(val, ScObjectType::Vec) {
return Err(ConversionError);
}
let vec = unsafe { Object::unchecked_from_val(val) };
let len: u32 = env.vec_len(vec).map_err(|_| ConversionError)?.try_into()?;
if len != $count {
return Err(ConversionError);
}
Ok((
$({
let idx: u32 = $idx;
let val = env.vec_get(vec, idx.into()).map_err(|_| ConversionError)?;
$typ::try_from_val(&env, &val).map_err(|_| ConversionError)?
},)*
))
}
}
impl<E: Env, $($typ),*> TryFromVal<E, ($($typ,)*)> for RawVal
where
$($typ: TryIntoVal<E, RawVal>),*
{
type Error = ConversionError;
fn try_from_val(env: &E, v: &($($typ,)*)) -> Result<Self, Self::Error> {
let vec = env.vec_new($count.into()).map_err(|_| ConversionError)?;
$(let vec = env.vec_push_back(vec, v.$idx.try_into_val(&env).map_err(|_| ConversionError)?).map_err(|_| ConversionError)?;)*
Ok(vec.to_raw())
}
}
impl<E: Env, $($typ),*, const N: usize> TryFromVal<E, [RawVal; N]> for ($($typ,)*)
where
$($typ: TryFromVal<E, RawVal>),*
{
type Error = ConversionError;
fn try_from_val(env: &E, val: &[RawVal; N]) -> Result<Self, Self::Error> {
Ok((
$({
$typ::try_from_val(&env, &val[$idx]).map_err(|_| ConversionError)?
},)*
))
}
}
impl<E: Env, $($typ),*, const N: usize> TryFromVal<E, ($($typ,)*)> for [RawVal; N]
where
$(RawVal: TryFromVal<E, $typ>),*
{
type Error = ConversionError;
fn try_from_val(env: &E, val: &($($typ,)*)) -> Result<Self, Self::Error> {
let mut arr = [RawVal::VOID; N];
$(arr[$idx] = val.$idx.try_into_val(env).map_err(|_| ConversionError)?;)*
Ok(arr)
}
}
};
}
impl_for_tuple! { 1_u32 1_usize T0 0 }
impl_for_tuple! { 2_u32 2_usize T0 0 T1 1 }
impl_for_tuple! { 3_u32 3_usize T0 0 T1 1 T2 2 }
impl_for_tuple! { 4_u32 4_usize T0 0 T1 1 T2 2 T3 3 }
impl_for_tuple! { 5_u32 5_usize T0 0 T1 1 T2 2 T3 3 T4 4 }
impl_for_tuple! { 6_u32 6_usize T0 0 T1 1 T2 2 T3 3 T4 4 T5 5 }
impl_for_tuple! { 7_u32 7_usize T0 0 T1 1 T2 2 T3 3 T4 4 T5 5 T6 6 }
impl_for_tuple! { 8_u32 8_usize T0 0 T1 1 T2 2 T3 3 T4 4 T5 5 T6 6 T7 7 }
impl_for_tuple! { 9_u32 9_usize T0 0 T1 1 T2 2 T3 3 T4 4 T5 5 T6 6 T7 7 T8 8 }
impl_for_tuple! { 10_u32 10_usize T0 0 T1 1 T2 2 T3 3 T4 4 T5 5 T6 6 T7 7 T8 8 T9 9 }
impl_for_tuple! { 11_u32 11_usize T0 0 T1 1 T2 2 T3 3 T4 4 T5 5 T6 6 T7 7 T8 8 T9 9 T10 10 }
impl_for_tuple! { 12_u32 12_usize T0 0 T1 1 T2 2 T3 3 T4 4 T5 5 T6 6 T7 7 T8 8 T9 9 T10 10 T11 11 }
impl_for_tuple! { 13_u32 13_usize T0 0 T1 1 T2 2 T3 3 T4 4 T5 5 T6 6 T7 7 T8 8 T9 9 T10 10 T11 11 T12 12 }
impl<E: Env> TryFromVal<E, [RawVal; 0]> for () {
type Error = ConversionError;
fn try_from_val(_env: &E, _val: &[RawVal; 0]) -> Result<Self, Self::Error> {
Ok(())
}
}
impl<E: Env> TryFromVal<E, ()> for [RawVal; 0] {
type Error = ConversionError;
fn try_from_val(_env: &E, _v: &()) -> Result<Self, Self::Error> {
Ok([RawVal::VOID; 0])
}
}