2021-09-13 17:43:54 +00:00
|
|
|
use std::collections::{BTreeMap, HashMap, VecDeque};
|
2021-03-12 14:57:46 +00:00
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
use crate::abi::{
|
2021-10-29 12:29:35 +00:00
|
|
|
error::{bail, format_err, ParseError, Result},
|
|
|
|
param_type::Reader,
|
|
|
|
struct_def::{FieldType, StructFieldType},
|
|
|
|
Abi, Constructor, Event, EventParam, Function, Param, ParamType, SolStruct, StateMutability,
|
2021-03-12 14:57:46 +00:00
|
|
|
};
|
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
/// A parser that turns a "human readable abi" into a `Abi`
|
|
|
|
pub struct AbiParser {
|
|
|
|
/// solidity structs
|
2021-03-16 19:37:19 +00:00
|
|
|
pub structs: HashMap<String, SolStruct>,
|
2021-03-15 11:49:28 +00:00
|
|
|
/// solidity structs as tuples
|
2021-03-16 19:37:19 +00:00
|
|
|
pub struct_tuples: HashMap<String, Vec<ParamType>>,
|
2021-10-29 12:29:35 +00:00
|
|
|
/// (function name, param name) -> struct which are the identifying properties we get the name
|
|
|
|
/// from ethabi.
|
2021-10-02 14:34:01 +00:00
|
|
|
pub function_params: HashMap<(String, String), String>,
|
|
|
|
/// (function name) -> Vec<structs> all structs the function returns
|
|
|
|
pub outputs: HashMap<String, Vec<String>>,
|
2021-03-15 11:49:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl AbiParser {
|
|
|
|
/// Parses a "human readable abi" string
|
2021-03-16 19:37:19 +00:00
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```
|
2021-08-28 21:06:29 +00:00
|
|
|
/// # use ethers_core::abi::AbiParser;
|
2021-03-16 19:37:19 +00:00
|
|
|
/// let abi = AbiParser::default().parse_str(r#"[
|
|
|
|
/// function setValue(string)
|
|
|
|
/// function getValue() external view returns (string)
|
|
|
|
/// event ValueChanged(address indexed author, string oldValue, string newValue)
|
|
|
|
/// ]"#).unwrap();
|
|
|
|
/// ```
|
|
|
|
pub fn parse_str(&mut self, s: &str) -> Result<Abi> {
|
|
|
|
self.parse(
|
2021-10-29 12:29:35 +00:00
|
|
|
&s.trim().trim_start_matches('[').trim_end_matches(']').lines().collect::<Vec<_>>(),
|
2021-03-16 19:37:19 +00:00
|
|
|
)
|
2020-10-29 07:48:24 +00:00
|
|
|
}
|
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
/// Parses a "human readable abi" string vector
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
/// ```
|
2021-08-28 21:06:29 +00:00
|
|
|
/// use ethers_core::abi::AbiParser;
|
2021-03-15 11:49:28 +00:00
|
|
|
///
|
|
|
|
/// let abi = AbiParser::default().parse(&[
|
|
|
|
/// "function x() external view returns (uint256)",
|
|
|
|
/// ]).unwrap();
|
|
|
|
/// ```
|
2021-03-16 19:37:19 +00:00
|
|
|
pub fn parse(&mut self, input: &[&str]) -> Result<Abi> {
|
2021-03-15 11:49:28 +00:00
|
|
|
// parse struct first
|
2021-03-16 19:37:19 +00:00
|
|
|
let mut abi = Abi {
|
|
|
|
constructor: None,
|
2021-09-13 17:43:54 +00:00
|
|
|
functions: BTreeMap::new(),
|
|
|
|
events: BTreeMap::new(),
|
2021-12-16 22:24:10 +00:00
|
|
|
errors: BTreeMap::new(),
|
2021-03-16 19:37:19 +00:00
|
|
|
receive: false,
|
|
|
|
fallback: false,
|
|
|
|
};
|
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
let (structs, types): (Vec<_>, Vec<_>) = input
|
|
|
|
.iter()
|
|
|
|
.map(|s| escape_quotes(s))
|
2021-03-16 19:37:19 +00:00
|
|
|
.map(str::trim)
|
|
|
|
.filter(|s| !s.is_empty())
|
2021-03-15 11:49:28 +00:00
|
|
|
.partition(|s| s.starts_with("struct"));
|
|
|
|
|
|
|
|
for sol in structs {
|
|
|
|
let s = SolStruct::parse(sol)?;
|
|
|
|
if self.structs.contains_key(s.name()) {
|
|
|
|
bail!("Duplicate struct declaration for struct `{}`", s.name())
|
|
|
|
}
|
|
|
|
self.structs.insert(s.name().to_string(), s);
|
|
|
|
}
|
|
|
|
self.substitute_structs()?;
|
|
|
|
|
|
|
|
for mut line in types {
|
|
|
|
line = line.trim_start();
|
2021-09-28 10:24:00 +00:00
|
|
|
if line.starts_with("event") {
|
2021-03-15 11:49:28 +00:00
|
|
|
let event = self.parse_event(line)?;
|
2021-10-29 12:29:35 +00:00
|
|
|
abi.events.entry(event.name.clone()).or_default().push(event);
|
2021-03-15 11:49:28 +00:00
|
|
|
} else if line.starts_with("constructor") {
|
2021-10-02 14:34:01 +00:00
|
|
|
let inputs = self
|
|
|
|
.constructor_inputs(line)?
|
|
|
|
.into_iter()
|
|
|
|
.map(|(input, struct_name)| {
|
|
|
|
if let Some(struct_name) = struct_name {
|
|
|
|
// keep track of the user defined struct of that param
|
|
|
|
self.function_params.insert(
|
|
|
|
("constructor".to_string(), input.name.clone()),
|
|
|
|
struct_name,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
input
|
|
|
|
})
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
abi.constructor = Some(Constructor { inputs });
|
2021-03-15 11:49:28 +00:00
|
|
|
} else {
|
2021-09-28 10:24:00 +00:00
|
|
|
// function may have shorthand declaration, so it won't start with "function"
|
|
|
|
let function = match self.parse_function(line) {
|
|
|
|
Ok(function) => function,
|
|
|
|
Err(_) => bail!("Illegal abi `{}`", line),
|
|
|
|
};
|
2021-10-29 12:29:35 +00:00
|
|
|
abi.functions.entry(function.name.clone()).or_default().push(function);
|
2021-03-15 11:49:28 +00:00
|
|
|
}
|
|
|
|
}
|
2021-03-16 19:37:19 +00:00
|
|
|
Ok(abi)
|
2021-03-15 11:49:28 +00:00
|
|
|
}
|
2020-10-29 07:48:24 +00:00
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
/// Substitutes any other struct references within structs with tuples
|
|
|
|
fn substitute_structs(&mut self) -> Result<()> {
|
|
|
|
let mut unresolved = self.structs.keys().collect::<VecDeque<_>>();
|
|
|
|
let mut sequential_retries = 0;
|
|
|
|
while let Some(name) = unresolved.pop_front() {
|
|
|
|
let mut resolved = true;
|
|
|
|
let sol = &self.structs[name];
|
|
|
|
let mut tuple = Vec::with_capacity(sol.fields().len());
|
|
|
|
for field in sol.fields() {
|
|
|
|
match field.r#type() {
|
|
|
|
FieldType::Elementary(param) => tuple.push(param.clone()),
|
|
|
|
FieldType::Struct(ty) => {
|
|
|
|
if let Some(param) = self.struct_tuples.get(ty.name()).cloned() {
|
2021-08-16 07:29:44 +00:00
|
|
|
tuple.push(ty.as_param(ParamType::Tuple(param)))
|
2021-03-15 11:49:28 +00:00
|
|
|
} else {
|
|
|
|
resolved = false;
|
2021-10-29 12:29:35 +00:00
|
|
|
break
|
2021-03-15 11:49:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
FieldType::Mapping(_) => {
|
|
|
|
bail!(
|
|
|
|
"mappings are not allowed as params in public functions of struct `{}`",
|
|
|
|
sol.name()
|
|
|
|
)
|
|
|
|
}
|
2021-03-12 14:57:46 +00:00
|
|
|
}
|
2021-03-15 11:49:28 +00:00
|
|
|
}
|
|
|
|
if resolved {
|
|
|
|
sequential_retries = 0;
|
|
|
|
self.struct_tuples.insert(sol.name().to_string(), tuple);
|
|
|
|
} else {
|
|
|
|
sequential_retries += 1;
|
|
|
|
if sequential_retries > unresolved.len() {
|
|
|
|
bail!("No struct definition found for struct `{}`", name)
|
|
|
|
}
|
|
|
|
unresolved.push_back(name);
|
2021-03-12 14:57:46 +00:00
|
|
|
}
|
|
|
|
}
|
2021-03-15 11:49:28 +00:00
|
|
|
Ok(())
|
2021-03-12 14:57:46 +00:00
|
|
|
}
|
2020-10-29 07:48:24 +00:00
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
/// Link additional structs for parsing
|
|
|
|
pub fn with_structs(structs: Vec<SolStruct>) -> Self {
|
|
|
|
Self {
|
2021-10-29 12:29:35 +00:00
|
|
|
structs: structs.into_iter().map(|s| (s.name().to_string(), s)).collect(),
|
2021-03-15 11:49:28 +00:00
|
|
|
struct_tuples: HashMap::new(),
|
2021-10-02 14:34:01 +00:00
|
|
|
function_params: Default::default(),
|
|
|
|
outputs: Default::default(),
|
2021-03-15 11:49:28 +00:00
|
|
|
}
|
2021-03-12 14:57:46 +00:00
|
|
|
}
|
2020-10-29 07:48:24 +00:00
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
/// Parses a solidity event declaration from `event <name> (args*) anonymous?`
|
2021-03-15 11:59:52 +00:00
|
|
|
pub fn parse_event(&self, s: &str) -> Result<Event> {
|
2021-03-15 11:49:28 +00:00
|
|
|
let mut event = s.trim();
|
|
|
|
if !event.starts_with("event ") {
|
|
|
|
bail!("Not an event `{}`", s)
|
|
|
|
}
|
|
|
|
event = &event[5..];
|
2020-10-29 07:48:24 +00:00
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
let name = parse_identifier(&mut event)?;
|
2021-03-12 14:57:46 +00:00
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
let mut chars = event.chars();
|
|
|
|
|
|
|
|
loop {
|
|
|
|
match chars.next() {
|
|
|
|
None => bail!("Expected event"),
|
|
|
|
Some('(') => {
|
|
|
|
event = chars.as_str().trim();
|
|
|
|
let mut anonymous = false;
|
|
|
|
if event.ends_with("anonymous") {
|
|
|
|
anonymous = true;
|
|
|
|
event = event[..event.len() - 9].trim_end();
|
|
|
|
}
|
|
|
|
event = event
|
|
|
|
.trim()
|
|
|
|
.strip_suffix(')')
|
|
|
|
.ok_or_else(|| format_err!("Expected closing `)` in `{}`", s))?;
|
|
|
|
|
|
|
|
let inputs = if event.is_empty() {
|
|
|
|
Vec::new()
|
|
|
|
} else {
|
|
|
|
event
|
|
|
|
.split(',')
|
|
|
|
.map(|e| self.parse_event_arg(e))
|
|
|
|
.collect::<Result<Vec<_>, _>>()?
|
|
|
|
};
|
2021-10-29 12:29:35 +00:00
|
|
|
return Ok(Event { name, inputs, anonymous })
|
2021-03-15 11:49:28 +00:00
|
|
|
}
|
2021-10-29 12:29:35 +00:00
|
|
|
Some(' ') | Some('\t') => continue,
|
2021-03-15 11:49:28 +00:00
|
|
|
Some(c) => {
|
|
|
|
bail!("Illegal char `{}` at `{}`", c, s)
|
2021-03-12 14:57:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
/// Parse a single event param
|
|
|
|
fn parse_event_arg(&self, input: &str) -> Result<EventParam> {
|
|
|
|
let mut iter = input.trim().rsplitn(3, is_whitespace);
|
|
|
|
let mut indexed = false;
|
2021-10-29 12:29:35 +00:00
|
|
|
let mut name =
|
|
|
|
iter.next().ok_or_else(|| format_err!("Empty event param at `{}`", input))?;
|
2021-03-15 11:49:28 +00:00
|
|
|
|
|
|
|
let type_str;
|
|
|
|
if let Some(mid) = iter.next() {
|
|
|
|
if let Some(ty) = iter.next() {
|
|
|
|
if mid != "indexed" {
|
|
|
|
bail!("Expected indexed keyword at `{}`", input)
|
|
|
|
}
|
2021-03-12 14:57:46 +00:00
|
|
|
indexed = true;
|
2021-03-15 11:49:28 +00:00
|
|
|
type_str = ty;
|
|
|
|
} else {
|
|
|
|
if name == "indexed" {
|
|
|
|
indexed = true;
|
|
|
|
name = "";
|
|
|
|
}
|
|
|
|
type_str = mid;
|
2021-03-12 14:57:46 +00:00
|
|
|
}
|
2021-03-15 11:49:28 +00:00
|
|
|
} else {
|
|
|
|
type_str = name;
|
|
|
|
name = "";
|
2021-03-12 14:57:46 +00:00
|
|
|
}
|
2021-03-15 11:49:28 +00:00
|
|
|
|
2021-10-29 12:29:35 +00:00
|
|
|
Ok(EventParam { name: name.to_string(), indexed, kind: self.parse_type(type_str)?.0 })
|
2021-03-15 11:49:28 +00:00
|
|
|
}
|
|
|
|
|
2021-03-15 11:59:52 +00:00
|
|
|
pub fn parse_function(&mut self, s: &str) -> Result<Function> {
|
2021-03-15 11:49:28 +00:00
|
|
|
let mut input = s.trim();
|
2021-09-28 10:24:00 +00:00
|
|
|
let shorthand = !input.starts_with("function ");
|
|
|
|
|
|
|
|
if !shorthand {
|
|
|
|
input = &input[8..];
|
2021-03-15 11:49:28 +00:00
|
|
|
}
|
2021-09-28 10:24:00 +00:00
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
let name = parse_identifier(&mut input)?;
|
2021-09-28 10:24:00 +00:00
|
|
|
input = input
|
|
|
|
.strip_prefix('(')
|
|
|
|
.ok_or_else(|| format_err!("Expected input args parentheses at `{}`", s))?;
|
2021-03-15 11:49:28 +00:00
|
|
|
|
2021-09-28 10:24:00 +00:00
|
|
|
let (input_args_modifiers, output_args) = match input.rsplit_once('(') {
|
2021-09-29 14:28:30 +00:00
|
|
|
Some((first, second)) => (first, Some(second)),
|
|
|
|
None => (input, None),
|
|
|
|
};
|
2021-03-15 11:49:28 +00:00
|
|
|
|
2021-09-28 10:24:00 +00:00
|
|
|
let mut input_args_modifiers_iter = input_args_modifiers
|
|
|
|
.trim_end()
|
|
|
|
.strip_suffix(" returns")
|
|
|
|
.unwrap_or(input_args_modifiers)
|
|
|
|
.splitn(2, ')');
|
2021-03-15 11:49:28 +00:00
|
|
|
|
2021-09-28 10:24:00 +00:00
|
|
|
let input_args = match input_args_modifiers_iter
|
|
|
|
.next()
|
|
|
|
.ok_or_else(|| format_err!("Expected input args parentheses at `{}`", s))?
|
|
|
|
{
|
|
|
|
"" => None,
|
|
|
|
input_params_args => Some(input_params_args),
|
|
|
|
};
|
|
|
|
let modifiers = match input_args_modifiers_iter
|
|
|
|
.next()
|
|
|
|
.ok_or_else(|| format_err!("Expected input args parentheses at `{}`", s))?
|
|
|
|
{
|
|
|
|
"" => None,
|
|
|
|
modifiers => Some(modifiers),
|
|
|
|
};
|
2021-03-15 11:49:28 +00:00
|
|
|
|
2021-09-28 10:24:00 +00:00
|
|
|
let inputs = if let Some(params) = input_args {
|
|
|
|
self.parse_params(params)?
|
2021-10-02 14:34:01 +00:00
|
|
|
.into_iter()
|
|
|
|
.map(|(input, struct_name)| {
|
|
|
|
if let Some(struct_name) = struct_name {
|
|
|
|
// keep track of the user defined struct of that param
|
|
|
|
self.function_params
|
|
|
|
.insert((name.clone(), input.name.clone()), struct_name);
|
|
|
|
}
|
|
|
|
input
|
|
|
|
})
|
|
|
|
.collect()
|
2021-03-15 11:49:28 +00:00
|
|
|
} else {
|
2021-09-28 10:24:00 +00:00
|
|
|
Vec::new()
|
|
|
|
};
|
2021-03-15 11:49:28 +00:00
|
|
|
|
2021-09-28 10:24:00 +00:00
|
|
|
let outputs = if let Some(params) = output_args {
|
2021-03-15 11:49:28 +00:00
|
|
|
let params = params
|
|
|
|
.trim()
|
2021-09-28 10:24:00 +00:00
|
|
|
.strip_suffix(')')
|
|
|
|
.ok_or_else(|| format_err!("Expected output args parentheses at `{}`", s))?;
|
2021-10-02 14:34:01 +00:00
|
|
|
let output_params = self.parse_params(params)?;
|
|
|
|
let mut outputs = Vec::with_capacity(output_params.len());
|
|
|
|
let mut output_types = Vec::new();
|
|
|
|
|
|
|
|
for (output, struct_name) in output_params {
|
|
|
|
if let Some(struct_name) = struct_name {
|
|
|
|
// keep track of the user defined struct of that param
|
|
|
|
output_types.push(struct_name);
|
|
|
|
}
|
|
|
|
outputs.push(output);
|
|
|
|
}
|
|
|
|
self.outputs.insert(name.clone(), output_types);
|
|
|
|
outputs
|
2021-03-15 11:49:28 +00:00
|
|
|
} else {
|
|
|
|
Vec::new()
|
|
|
|
};
|
|
|
|
|
|
|
|
let state_mutability = modifiers.map(detect_state_mutability).unwrap_or_default();
|
|
|
|
|
2021-10-18 10:28:38 +00:00
|
|
|
Ok(
|
|
|
|
#[allow(deprecated)]
|
2022-01-13 00:58:11 +00:00
|
|
|
Function { name, inputs, outputs, state_mutability, constant: None },
|
2021-10-18 10:28:38 +00:00
|
|
|
)
|
2021-03-12 14:57:46 +00:00
|
|
|
}
|
2020-10-29 07:48:24 +00:00
|
|
|
|
2021-10-02 14:34:01 +00:00
|
|
|
fn parse_params(&self, s: &str) -> Result<Vec<(Param, Option<String>)>> {
|
2021-03-15 11:49:28 +00:00
|
|
|
s.split(',')
|
|
|
|
.filter(|s| !s.is_empty())
|
|
|
|
.map(|s| self.parse_param(s))
|
|
|
|
.collect::<Result<Vec<_>, _>>()
|
2021-03-12 14:57:46 +00:00
|
|
|
}
|
2021-03-15 11:49:28 +00:00
|
|
|
|
2021-10-29 12:29:35 +00:00
|
|
|
/// Returns the `ethabi` `ParamType` for the function parameter and the aliased struct type, if
|
|
|
|
/// it is a user defined struct
|
2021-10-02 14:34:01 +00:00
|
|
|
fn parse_type(&self, type_str: &str) -> Result<(ParamType, Option<String>)> {
|
2021-03-15 11:49:28 +00:00
|
|
|
if let Ok(kind) = Reader::read(type_str) {
|
2021-10-02 14:34:01 +00:00
|
|
|
Ok((kind, None))
|
2021-03-15 11:49:28 +00:00
|
|
|
} else {
|
|
|
|
// try struct instead
|
|
|
|
if let Ok(field) = StructFieldType::parse(type_str) {
|
|
|
|
let struct_ty = field
|
|
|
|
.as_struct()
|
|
|
|
.ok_or_else(|| format_err!("Expected struct type `{}`", type_str))?;
|
2021-10-02 14:34:01 +00:00
|
|
|
let name = struct_ty.name();
|
2021-03-15 11:49:28 +00:00
|
|
|
let tuple = self
|
|
|
|
.struct_tuples
|
2021-10-02 14:34:01 +00:00
|
|
|
.get(name)
|
2021-03-15 11:49:28 +00:00
|
|
|
.cloned()
|
|
|
|
.map(ParamType::Tuple)
|
|
|
|
.ok_or_else(|| format_err!("Unknown struct `{}`", struct_ty.name()))?;
|
|
|
|
|
2021-08-16 07:29:44 +00:00
|
|
|
if let Some(field) = field.as_struct() {
|
2021-10-02 14:34:01 +00:00
|
|
|
Ok((field.as_param(tuple), Some(name.to_string())))
|
2021-08-16 07:29:44 +00:00
|
|
|
} else {
|
|
|
|
bail!("Expected struct type")
|
2021-03-15 11:49:28 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
bail!("Failed determine event type `{}`", type_str)
|
|
|
|
}
|
|
|
|
}
|
2021-03-12 14:57:46 +00:00
|
|
|
}
|
|
|
|
|
2021-03-15 11:59:52 +00:00
|
|
|
pub fn parse_constructor(&self, s: &str) -> Result<Constructor> {
|
2021-10-29 12:29:35 +00:00
|
|
|
let inputs = self.constructor_inputs(s)?.into_iter().map(|s| s.0).collect();
|
2021-10-02 14:34:01 +00:00
|
|
|
Ok(Constructor { inputs })
|
|
|
|
}
|
|
|
|
|
|
|
|
fn constructor_inputs(&self, s: &str) -> Result<Vec<(Param, Option<String>)>> {
|
2021-03-15 11:49:28 +00:00
|
|
|
let mut input = s.trim();
|
|
|
|
if !input.starts_with("constructor") {
|
|
|
|
bail!("Not a constructor `{}`", input)
|
|
|
|
}
|
|
|
|
input = input[11..]
|
|
|
|
.trim_start()
|
|
|
|
.strip_prefix('(')
|
|
|
|
.ok_or_else(|| format_err!("Expected leading `(` in `{}`", s))?;
|
2021-03-12 14:57:46 +00:00
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
let params = input
|
|
|
|
.rsplitn(2, ')')
|
|
|
|
.last()
|
|
|
|
.ok_or_else(|| format_err!("Expected closing `)` in `{}`", s))?;
|
2021-03-12 14:57:46 +00:00
|
|
|
|
2021-10-02 14:34:01 +00:00
|
|
|
self.parse_params(params)
|
2021-03-15 11:49:28 +00:00
|
|
|
}
|
|
|
|
|
2021-10-02 14:34:01 +00:00
|
|
|
fn parse_param(&self, param: &str) -> Result<(Param, Option<String>)> {
|
2021-03-15 11:49:28 +00:00
|
|
|
let mut iter = param.trim().rsplitn(3, is_whitespace);
|
|
|
|
|
2021-10-29 12:29:35 +00:00
|
|
|
let mut name = iter.next().ok_or(ParseError::ParseError(super::Error::InvalidData))?;
|
2020-10-29 07:48:24 +00:00
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
let type_str;
|
|
|
|
if let Some(ty) = iter.last() {
|
|
|
|
if name == "memory" || name == "calldata" {
|
|
|
|
name = "";
|
|
|
|
}
|
|
|
|
type_str = ty;
|
|
|
|
} else {
|
|
|
|
type_str = name;
|
|
|
|
name = "";
|
|
|
|
}
|
2021-10-02 14:34:01 +00:00
|
|
|
let (kind, user_struct) = self.parse_type(type_str)?;
|
2021-10-29 12:29:35 +00:00
|
|
|
Ok((Param { name: name.to_string(), kind, internal_type: None }, user_struct))
|
2021-03-12 14:57:46 +00:00
|
|
|
}
|
2021-03-15 11:49:28 +00:00
|
|
|
}
|
2021-03-12 14:57:46 +00:00
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
impl Default for AbiParser {
|
|
|
|
fn default() -> Self {
|
|
|
|
Self::with_structs(Vec::new())
|
|
|
|
}
|
|
|
|
}
|
2021-03-12 14:57:46 +00:00
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
/// Parses a "human readable abi" string vector
|
|
|
|
///
|
|
|
|
/// ```
|
2021-08-28 21:06:29 +00:00
|
|
|
/// use ethers_core::abi::parse_abi;
|
2021-03-15 11:49:28 +00:00
|
|
|
///
|
|
|
|
/// let abi = parse_abi(&[
|
|
|
|
/// "function x() external view returns (uint256)",
|
|
|
|
/// ]).unwrap();
|
|
|
|
/// ```
|
|
|
|
pub fn parse(input: &[&str]) -> Result<Abi> {
|
|
|
|
AbiParser::default().parse(input)
|
|
|
|
}
|
2021-03-12 14:57:46 +00:00
|
|
|
|
2021-03-16 19:37:19 +00:00
|
|
|
/// Parses a "human readable abi" string
|
|
|
|
///
|
|
|
|
/// See also `AbiParser::parse_str`
|
|
|
|
pub fn parse_str(input: &str) -> Result<Abi> {
|
|
|
|
AbiParser::default().parse_str(input)
|
|
|
|
}
|
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
/// Parses an identifier like event or function name
|
|
|
|
pub(crate) fn parse_identifier(input: &mut &str) -> Result<String> {
|
|
|
|
let mut chars = input.trim_start().chars();
|
|
|
|
let mut name = String::new();
|
2021-10-29 12:29:35 +00:00
|
|
|
let c = chars.next().ok_or_else(|| format_err!("Empty identifier in `{}`", input))?;
|
2021-03-15 11:49:28 +00:00
|
|
|
if is_first_ident_char(c) {
|
|
|
|
name.push(c);
|
|
|
|
loop {
|
|
|
|
match chars.clone().next() {
|
|
|
|
Some(c) if is_ident_char(c) => {
|
|
|
|
chars.next();
|
|
|
|
name.push(c);
|
|
|
|
}
|
|
|
|
_ => break,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if name.is_empty() {
|
2021-10-29 12:29:35 +00:00
|
|
|
return Err(ParseError::ParseError(super::Error::InvalidName(input.to_string())))
|
2021-03-15 11:49:28 +00:00
|
|
|
}
|
|
|
|
*input = chars.as_str();
|
|
|
|
Ok(name)
|
2021-03-12 14:57:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn detect_state_mutability(s: &str) -> StateMutability {
|
|
|
|
if s.contains("pure") {
|
|
|
|
StateMutability::Pure
|
|
|
|
} else if s.contains("view") {
|
|
|
|
StateMutability::View
|
|
|
|
} else if s.contains("payable") {
|
|
|
|
StateMutability::Payable
|
|
|
|
} else {
|
|
|
|
StateMutability::NonPayable
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
pub(crate) fn is_first_ident_char(c: char) -> bool {
|
2021-03-12 14:57:46 +00:00
|
|
|
matches!(c, 'a'..='z' | 'A'..='Z' | '_')
|
|
|
|
}
|
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
pub(crate) fn is_ident_char(c: char) -> bool {
|
2021-03-12 14:57:46 +00:00
|
|
|
matches!(c, 'a'..='z' | 'A'..='Z' | '0'..='9' | '_')
|
|
|
|
}
|
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
pub(crate) fn is_whitespace(c: char) -> bool {
|
2021-03-12 14:57:46 +00:00
|
|
|
matches!(c, ' ' | '\t')
|
|
|
|
}
|
|
|
|
|
|
|
|
fn escape_quotes(input: &str) -> &str {
|
|
|
|
input.trim_matches(is_whitespace).trim_matches('\"')
|
2020-10-29 07:48:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parses_approve() {
|
|
|
|
let fn_str = "function approve(address _spender, uint256 value) external returns(bool)";
|
2021-03-15 11:49:28 +00:00
|
|
|
let parsed = AbiParser::default().parse_function(fn_str).unwrap();
|
2020-10-29 07:48:24 +00:00
|
|
|
assert_eq!(parsed.name, "approve");
|
|
|
|
assert_eq!(parsed.inputs[0].name, "_spender");
|
|
|
|
assert_eq!(parsed.inputs[0].kind, ParamType::Address,);
|
|
|
|
assert_eq!(parsed.inputs[1].name, "value");
|
|
|
|
assert_eq!(parsed.inputs[1].kind, ParamType::Uint(256),);
|
|
|
|
assert_eq!(parsed.outputs[0].name, "");
|
|
|
|
assert_eq!(parsed.outputs[0].kind, ParamType::Bool);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parses_function_arguments_return() {
|
|
|
|
let fn_str = "function foo(uint32[] memory x) external view returns (address)";
|
2021-03-15 11:49:28 +00:00
|
|
|
let parsed = AbiParser::default().parse_function(fn_str).unwrap();
|
2020-10-29 07:48:24 +00:00
|
|
|
assert_eq!(parsed.name, "foo");
|
|
|
|
assert_eq!(parsed.inputs[0].name, "x");
|
2021-10-29 12:29:35 +00:00
|
|
|
assert_eq!(parsed.inputs[0].kind, ParamType::Array(Box::new(ParamType::Uint(32))));
|
2020-10-29 07:48:24 +00:00
|
|
|
assert_eq!(parsed.outputs[0].name, "");
|
|
|
|
assert_eq!(parsed.outputs[0].kind, ParamType::Address);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parses_function_empty() {
|
|
|
|
let fn_str = "function foo()";
|
2021-03-15 11:49:28 +00:00
|
|
|
let parsed = AbiParser::default().parse_function(fn_str).unwrap();
|
2020-10-29 07:48:24 +00:00
|
|
|
assert_eq!(parsed.name, "foo");
|
|
|
|
assert!(parsed.inputs.is_empty());
|
|
|
|
assert!(parsed.outputs.is_empty());
|
|
|
|
}
|
|
|
|
|
2021-03-12 14:57:46 +00:00
|
|
|
#[test]
|
|
|
|
fn parses_function_payable() {
|
|
|
|
let fn_str = "function foo() public payable";
|
2021-03-15 11:49:28 +00:00
|
|
|
let parsed = AbiParser::default().parse_function(fn_str).unwrap();
|
2021-03-12 14:57:46 +00:00
|
|
|
assert_eq!(parsed.state_mutability, StateMutability::Payable);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parses_function_view() {
|
|
|
|
let fn_str = "function foo() external view";
|
2021-03-15 11:49:28 +00:00
|
|
|
let parsed = AbiParser::default().parse_function(fn_str).unwrap();
|
2021-03-12 14:57:46 +00:00
|
|
|
assert_eq!(parsed.state_mutability, StateMutability::View);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parses_function_pure() {
|
|
|
|
let fn_str = "function foo() pure";
|
2021-03-15 11:49:28 +00:00
|
|
|
let parsed = AbiParser::default().parse_function(fn_str).unwrap();
|
2021-03-12 14:57:46 +00:00
|
|
|
assert_eq!(parsed.state_mutability, StateMutability::Pure);
|
|
|
|
}
|
|
|
|
|
2020-10-29 07:48:24 +00:00
|
|
|
#[test]
|
|
|
|
fn parses_event() {
|
|
|
|
assert_eq!(
|
2021-03-15 11:49:28 +00:00
|
|
|
AbiParser::default()
|
2021-07-05 11:03:38 +00:00
|
|
|
.parse_event("event Foo (address indexed x, uint y, bytes32[] z)")
|
2021-03-15 11:49:28 +00:00
|
|
|
.unwrap(),
|
2020-10-29 07:48:24 +00:00
|
|
|
Event {
|
|
|
|
anonymous: false,
|
2021-03-15 11:49:28 +00:00
|
|
|
name: "Foo".to_string(),
|
2020-10-29 07:48:24 +00:00
|
|
|
inputs: vec![
|
2021-10-29 12:29:35 +00:00
|
|
|
EventParam { name: "x".to_string(), kind: ParamType::Address, indexed: true },
|
2020-10-29 07:48:24 +00:00
|
|
|
EventParam {
|
2021-03-15 11:49:28 +00:00
|
|
|
name: "y".to_string(),
|
2020-10-29 07:48:24 +00:00
|
|
|
kind: ParamType::Uint(256),
|
|
|
|
indexed: false,
|
|
|
|
},
|
|
|
|
EventParam {
|
2021-03-15 11:49:28 +00:00
|
|
|
name: "z".to_string(),
|
2020-10-29 07:48:24 +00:00
|
|
|
kind: ParamType::Array(Box::new(ParamType::FixedBytes(32))),
|
|
|
|
indexed: false,
|
|
|
|
},
|
2021-03-12 14:57:46 +00:00
|
|
|
],
|
2020-10-29 07:48:24 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parses_anonymous_event() {
|
|
|
|
assert_eq!(
|
2021-10-29 12:29:35 +00:00
|
|
|
AbiParser::default().parse_event("event Foo() anonymous").unwrap(),
|
|
|
|
Event { anonymous: true, name: "Foo".to_string(), inputs: vec![] }
|
2020-10-29 07:48:24 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-03-12 14:57:46 +00:00
|
|
|
#[test]
|
|
|
|
fn parses_unnamed_event() {
|
|
|
|
assert_eq!(
|
2021-10-29 12:29:35 +00:00
|
|
|
AbiParser::default().parse_event("event Foo(address)").unwrap(),
|
2021-03-12 14:57:46 +00:00
|
|
|
Event {
|
|
|
|
anonymous: false,
|
2021-03-15 11:49:28 +00:00
|
|
|
name: "Foo".to_string(),
|
2021-03-12 14:57:46 +00:00
|
|
|
inputs: vec![EventParam {
|
2021-03-15 11:49:28 +00:00
|
|
|
name: "".to_string(),
|
2021-03-12 14:57:46 +00:00
|
|
|
kind: ParamType::Address,
|
|
|
|
indexed: false,
|
|
|
|
}],
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn parses_unnamed_indexed_event() {
|
|
|
|
assert_eq!(
|
2021-10-29 12:29:35 +00:00
|
|
|
AbiParser::default().parse_event("event Foo(address indexed)").unwrap(),
|
2021-03-12 14:57:46 +00:00
|
|
|
Event {
|
|
|
|
anonymous: false,
|
2021-03-15 11:49:28 +00:00
|
|
|
name: "Foo".to_string(),
|
2021-03-12 14:57:46 +00:00
|
|
|
inputs: vec![EventParam {
|
2021-03-15 11:49:28 +00:00
|
|
|
name: "".to_string(),
|
2021-03-12 14:57:46 +00:00
|
|
|
kind: ParamType::Address,
|
|
|
|
indexed: true,
|
|
|
|
}],
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-10-29 07:48:24 +00:00
|
|
|
#[test]
|
|
|
|
fn parse_event_input() {
|
|
|
|
assert_eq!(
|
2021-10-29 12:29:35 +00:00
|
|
|
AbiParser::default().parse_event_arg("address indexed x").unwrap(),
|
|
|
|
EventParam { name: "x".to_string(), kind: ParamType::Address, indexed: true }
|
2020-10-29 07:48:24 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
assert_eq!(
|
2021-03-15 11:49:28 +00:00
|
|
|
AbiParser::default().parse_event_arg("address x").unwrap(),
|
2021-10-29 12:29:35 +00:00
|
|
|
EventParam { name: "x".to_string(), kind: ParamType::Address, indexed: false }
|
2020-10-29 07:48:24 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_parse_functions() {
|
|
|
|
[
|
|
|
|
"function foo(uint256[] memory x) external view returns (address)",
|
2021-09-28 10:24:00 +00:00
|
|
|
"function bar(uint256[] memory x) returns(address)",
|
2020-10-29 07:48:24 +00:00
|
|
|
"function bar(uint256[] memory x, uint32 y) returns (address, uint256)",
|
2020-10-30 10:15:34 +00:00
|
|
|
"function foo(address[] memory, bytes memory) returns (bytes memory)",
|
2020-10-29 07:48:24 +00:00
|
|
|
"function bar(uint256[] memory x)",
|
|
|
|
"function bar()",
|
2021-09-28 10:24:00 +00:00
|
|
|
"bar(uint256[] memory x)(address)",
|
|
|
|
"bar(uint256[] memory x, uint32 y)(address, uint256)",
|
|
|
|
"foo(address[] memory, bytes memory)(bytes memory)",
|
|
|
|
"bar(uint256[] memory x)()",
|
|
|
|
"bar()()",
|
2021-09-29 14:28:30 +00:00
|
|
|
"bar(uint256)",
|
|
|
|
"bar()",
|
2020-10-29 07:48:24 +00:00
|
|
|
]
|
|
|
|
.iter()
|
|
|
|
.for_each(|x| {
|
2021-03-15 11:49:28 +00:00
|
|
|
AbiParser::default().parse_function(x).unwrap();
|
2020-10-29 07:48:24 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-03-15 11:49:28 +00:00
|
|
|
#[test]
|
|
|
|
fn can_parse_structs_and_functions() {
|
|
|
|
let abi = &[
|
|
|
|
"struct Demo {bytes x; address payable d;}",
|
|
|
|
"struct Voter { uint weight; bool voted; address delegate; uint vote; }",
|
|
|
|
"event FireEvent(Voter v, NestedVoter2 n)",
|
|
|
|
"function foo(uint256[] memory x) external view returns (address)",
|
|
|
|
"function call(Voter memory voter) returns (address, uint256)",
|
2021-09-28 10:24:00 +00:00
|
|
|
"foo(uint256[] memory x)()",
|
|
|
|
"call(Voter memory voter)(address, uint256)",
|
2021-03-15 11:49:28 +00:00
|
|
|
"struct NestedVoter { Voter voter; bool voted; address delegate; uint vote; }",
|
|
|
|
"struct NestedVoter2 { NestedVoter[] voter; Voter[10] votes; address delegate; uint vote; }",
|
|
|
|
];
|
|
|
|
parse(abi).unwrap();
|
|
|
|
}
|
|
|
|
|
2020-10-30 10:15:34 +00:00
|
|
|
#[test]
|
|
|
|
fn can_parse_params() {
|
|
|
|
[
|
|
|
|
"address x",
|
|
|
|
"address",
|
|
|
|
"bytes memory y",
|
|
|
|
"bytes memory",
|
|
|
|
"bytes32[] memory",
|
|
|
|
"bytes32[] memory z",
|
|
|
|
]
|
|
|
|
.iter()
|
|
|
|
.for_each(|x| {
|
2021-03-15 11:49:28 +00:00
|
|
|
AbiParser::default().parse_param(x).unwrap();
|
2020-10-30 10:15:34 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-10-29 07:48:24 +00:00
|
|
|
#[test]
|
|
|
|
fn can_read_backslashes() {
|
|
|
|
parse(&[
|
|
|
|
"\"function setValue(string)\"",
|
2021-03-12 14:57:46 +00:00
|
|
|
"\"function getValue() external view returns(string)\"",
|
2020-10-29 07:48:24 +00:00
|
|
|
])
|
|
|
|
.unwrap();
|
|
|
|
}
|
2021-03-15 11:49:28 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_substitute_structs() {
|
|
|
|
let abi = parse(&[
|
|
|
|
"struct MyStruct {int y; address _addr;}",
|
|
|
|
"event FireEvent(MyStruct m, address indexed newOwner)",
|
|
|
|
])
|
|
|
|
.unwrap();
|
|
|
|
assert_eq!(
|
|
|
|
abi.events["FireEvent"][0].inputs.clone(),
|
|
|
|
vec![
|
|
|
|
EventParam {
|
|
|
|
name: "m".to_string(),
|
|
|
|
kind: ParamType::Tuple(vec![ParamType::Int(256), ParamType::Address]),
|
|
|
|
indexed: false
|
|
|
|
},
|
|
|
|
EventParam {
|
|
|
|
name: "newOwner".to_string(),
|
|
|
|
kind: ParamType::Address,
|
|
|
|
indexed: true
|
|
|
|
},
|
|
|
|
]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_substitute_array_structs() {
|
|
|
|
let abi = parse(&[
|
|
|
|
"struct MyStruct {int y; address _addr;}",
|
|
|
|
"event FireEvent(MyStruct[] m, MyStruct[10] m2)",
|
|
|
|
])
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
abi.events["FireEvent"][0].inputs.clone(),
|
|
|
|
vec![
|
|
|
|
EventParam {
|
|
|
|
name: "m".to_string(),
|
|
|
|
kind: ParamType::Array(Box::new(ParamType::Tuple(vec![
|
|
|
|
ParamType::Int(256),
|
|
|
|
ParamType::Address
|
|
|
|
]))),
|
|
|
|
indexed: false
|
|
|
|
},
|
|
|
|
EventParam {
|
|
|
|
name: "m2".to_string(),
|
|
|
|
kind: ParamType::FixedArray(
|
2021-10-29 12:29:35 +00:00
|
|
|
Box::new(ParamType::Tuple(vec![ParamType::Int(256), ParamType::Address])),
|
2021-03-15 11:49:28 +00:00
|
|
|
10
|
|
|
|
),
|
|
|
|
indexed: false
|
|
|
|
},
|
|
|
|
]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn can_substitute_nested_array_structs() {
|
|
|
|
let abi = parse(&[
|
|
|
|
"struct MyStruct {int y; address _addr;}",
|
|
|
|
"event FireEvent(MyStruct[] m, MyStructWrapper w)",
|
|
|
|
"struct MyStructWrapper {MyStruct y; int y; address _addr;}",
|
|
|
|
])
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
abi.events["FireEvent"][0].inputs.clone(),
|
|
|
|
vec![
|
|
|
|
EventParam {
|
|
|
|
name: "m".to_string(),
|
|
|
|
kind: ParamType::Array(Box::new(ParamType::Tuple(vec![
|
|
|
|
ParamType::Int(256),
|
|
|
|
ParamType::Address
|
|
|
|
]))),
|
|
|
|
indexed: false
|
|
|
|
},
|
|
|
|
EventParam {
|
|
|
|
name: "w".to_string(),
|
|
|
|
kind: ParamType::Tuple(vec![
|
|
|
|
ParamType::Tuple(vec![ParamType::Int(256), ParamType::Address]),
|
|
|
|
ParamType::Int(256),
|
|
|
|
ParamType::Address
|
|
|
|
]),
|
|
|
|
indexed: false
|
|
|
|
},
|
|
|
|
]
|
|
|
|
);
|
|
|
|
}
|
2020-10-29 07:48:24 +00:00
|
|
|
}
|