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
116
117
use std::string::ToString;
use crate::{
decode, encode, signature::short_signature, Bytes, Error, Param, ParamType, Result, StateMutability, Token,
};
use serde::{Deserialize, Serialize};
#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)]
pub struct Function {
#[serde(deserialize_with = "crate::util::sanitize_name::deserialize")]
pub name: String,
pub inputs: Vec<Param>,
pub outputs: Vec<Param>,
#[deprecated(note = "The constant attribute was removed in Solidity 0.5.0 and has been \
replaced with stateMutability. If parsing a JSON AST created with \
this version or later this value will always be false, which may be wrong.")]
#[serde(default)]
pub constant: bool,
#[serde(rename = "stateMutability", default)]
pub state_mutability: StateMutability,
}
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, StateMutability, Token};
use hex_literal::hex;
#[test]
fn test_function_encode_call() {
#[allow(deprecated)]
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,
state_mutability: StateMutability::Payable,
};
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);
}
}