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
use std::string::ToString;
use crate::{decode, encode, signature::short_signature, Bytes, Error, Param, ParamType, Result, Token};
use serde::Deserialize;
#[derive(Debug, Clone, PartialEq, Deserialize)]
pub struct Function {
pub name: String,
pub inputs: Vec<Param>,
pub outputs: Vec<Param>,
#[serde(default)]
pub constant: bool,
}
impl Function {
fn input_param_types(&self) -> Vec<ParamType> {
self.inputs.iter().map(|p| p.kind.clone()).collect()
}
fn output_param_types(&self) -> Vec<ParamType> {
self.outputs.iter().map(|p| p.kind.clone()).collect()
}
pub fn encode_input(&self, tokens: &[Token]) -> Result<Bytes> {
let params = self.input_param_types();
if !Token::types_check(tokens, ¶ms) {
return Err(Error::InvalidData);
}
let signed = short_signature(&self.name, ¶ms).to_vec();
let encoded = encode(tokens);
Ok(signed.into_iter().chain(encoded.into_iter()).collect())
}
pub fn decode_output(&self, data: &[u8]) -> Result<Vec<Token>> {
decode(&self.output_param_types(), &data)
}
pub fn decode_input(&self, data: &[u8]) -> Result<Vec<Token>> {
decode(&self.input_param_types(), &data)
}
pub fn signature(&self) -> String {
let inputs = self.inputs.iter().map(|p| p.kind.to_string()).collect::<Vec<_>>().join(",");
let outputs = self.outputs.iter().map(|p| p.kind.to_string()).collect::<Vec<_>>().join(",");
match (inputs.len(), outputs.len()) {
(_, 0) => format!("{}({})", self.name, inputs),
(_, _) => format!("{}({}):({})", self.name, inputs, outputs),
}
}
}
#[cfg(test)]
mod tests {
use crate::{Function, Param, ParamType, Token};
use hex_literal::hex;
#[test]
fn test_function_encode_call() {
let func = Function {
name: "baz".to_owned(),
inputs: vec![
Param { name: "a".to_owned(), kind: ParamType::Uint(32) },
Param { name: "b".to_owned(), kind: ParamType::Bool },
],
outputs: vec![],
constant: false,
};
let mut uint = [0u8; 32];
uint[31] = 69;
let encoded = func.encode_input(&[Token::Uint(uint.into()), Token::Bool(true)]).unwrap();
let expected = hex!("cdcd77c000000000000000000000000000000000000000000000000000000000000000450000000000000000000000000000000000000000000000000000000000000001").to_vec();
assert_eq!(encoded, expected);
}
}