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
use crate::{
    encode::Encode,
    into_type::IntoType,
    selector::Selector,
};

#[derive(Default)]
pub struct Builder<'a> {
    name: Option<&'a str>,
    selector: Selector,
    pub(super) params: Vec<(bool, Vec<u8>)>,
}

impl<'a> Builder<'a> {
    pub fn new() -> Self {
        Builder::default()
    }

    pub fn name(mut self, name: &'a str) -> Self {
        self.name = Some(name);
        self
    }

    pub fn push<F: Encode + IntoType>(mut self, value: F) -> Self {
        self.selector = self.selector.push::<F>();
        self.params.push((F::is_dynamic(), value.encode()));
        self
    }

    pub fn build(self) -> Vec<u8> {
        let name_offset = if let Some(_) = self.name { 4 } else { 0 };

        let sig = if let Some(name) = self.name {
            Some(self.selector.build(name))
        } else {
            None
        };

        let total_len = self.params.iter().map(|param| param.1.len()).sum::<usize>()
            + self
                .params
                .iter()
                .map(|param| param.0)
                .filter(|&param| param == true)
                .count()
                * 32;

        let mut buf: Vec<u8> = vec![0; total_len + name_offset];

        let mut offset: usize = self.params.len() * 32 + name_offset;

        for (index, (dynamic, bytes)) in self.params.into_iter().enumerate() {
            if dynamic {
                buf[index * 32 + 24 + name_offset..(index + 1) * 32 + name_offset]
                    .copy_from_slice(&(offset as u64).to_be_bytes());
                buf[offset..offset + bytes.len()].copy_from_slice(&bytes);
                offset += bytes.len()
            } else {
                buf[index * 32 + name_offset..(index + 1) * 32 + name_offset]
                    .copy_from_slice(&bytes);
            }
        }

        if let Some(sig) = sig {
            buf[0..4].copy_from_slice(&sig)
        }

        buf
    }
}