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
/// Trait for getting the arity of a value. You shouldn't need to impl this yourself.
pub trait Arity {
    /// Gets the arity of the type.
    ///
    /// Example
    /// ```rust
    /// use tuple_arity::Arity;
    /// 
    /// assert_eq!(0, <()>::arity());
    /// assert_eq!(1, <(u8,)>::arity());
    /// assert_eq!(2, <(u8, u8)>::arity());
    /// assert_eq!(3, <(u8, u8, u8)>::arity());
    /// assert_eq!(4, <(u8, u8, u8, u8)>::arity());
    /// ```
    fn arity() -> usize;
}

macro_rules! impl_tuple_arity {
    ($len:expr, $($tuple_arg:ident),*) => {
        impl<$($tuple_arg,)*> Arity for ($($tuple_arg,)*) {
            #[inline(always)]
            fn arity() -> usize {
                $len
            }
        }
    }
}

/// Gets the arity of the specified value.
///
/// # Example
/// ```rust
/// use tuple_arity::*;
/// assert_eq!(0, tuple_arity(&()));
/// assert_eq!(1, tuple_arity(&("foo",)));
/// assert_eq!(2, tuple_arity(&("foo", "bar")));
/// assert_eq!(3, tuple_arity(&("foo", "bar", "baz")));
/// ```
#[inline(always)]
pub fn tuple_arity<T: Arity>(_: &T) -> usize {
    T::arity()
}


impl_tuple_arity!(0,);
impl_tuple_arity!(1, T1);
impl_tuple_arity!(2, T1, T2);
impl_tuple_arity!(3, T1, T2, T3);
impl_tuple_arity!(4, T1, T2, T3, T4);
impl_tuple_arity!(5, T1, T2, T3, T4, T5);
impl_tuple_arity!(6, T1, T2, T3, T4, T5, T6);
impl_tuple_arity!(7, T1, T2, T3, T4, T5, T6, T7);
impl_tuple_arity!(8, T1, T2, T3, T4, T5, T6, T7, T8);
impl_tuple_arity!(9, T1, T2, T3, T4, T5, T6, T7, T8, T9);
impl_tuple_arity!(10, T1, T2, T3, T4, T5, T6, T7, T8, T9, T10);
impl_tuple_arity!(11, T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11);
impl_tuple_arity!(12, T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12);