Skip to content

Commit

Permalink
Merge pull request #12 from ethcore/serde-derive
Browse files Browse the repository at this point in the history
Bumping serde
  • Loading branch information
debris authored Feb 9, 2017
2 parents 3b7284b + 9ac5a62 commit 9ef6d92
Show file tree
Hide file tree
Showing 10 changed files with 375 additions and 508 deletions.
184 changes: 37 additions & 147 deletions Cargo.lock

Large diffs are not rendered by default.

15 changes: 6 additions & 9 deletions Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -2,30 +2,27 @@
name = "ethabi"
version = "0.2.3"
authors = ["debris <[email protected]>"]
build = "build.rs"
homepage = "https://github.com/ethcore/ethabi"
license = "MIT"
keywords = ["ethereum", "eth", "abi", "solidity", "cli"]
description = "Easy to use conversion of ethereum contract calls to bytecode."

[dependencies]
rustc-serialize = "0.3"
serde = "0.8"
serde_json = "0.8"
serde = "0.9"
serde_json = "0.9"
serde_derive = "0.9"
tiny-keccak = "1.0"
serde_macros = { version = "0.8.0", optional = true }
clippy = { version = "0.0.63", optional = true }
docopt = { version = "0.6", optional = true }

[build-dependencies]
serde_codegen = { version = "0.8.0", optional = true }

[features]
default = ["serde_codegen", "cli"]
nightly = ["serde_macros"]
cli = ["docopt"]

[lib]

[[bin]]
name = "ethabi"
path = "src/bin/main.rs"
doc = false

25 changes: 0 additions & 25 deletions build.rs

This file was deleted.

293 changes: 293 additions & 0 deletions src/bin/ethbin.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,293 @@
extern crate docopt;
extern crate rustc_serialize;
extern crate ethabi;

mod error;

use std::fs::File;
use std::io::Read;
use std::env;
use docopt::Docopt;
use rustc_serialize::hex::{ToHex, FromHex};
use ethabi::spec::param_type::{ParamType, Reader};
use ethabi::token::{Token, Tokenizer, StrictTokenizer, LenientTokenizer, TokenFromHex};
use ethabi::{Encoder, Decoder, Contract, Function, Event, Interface};
use error::Error;

pub const ETHABI: &'static str = r#"
Ethereum ABI coder.
Copyright 2016-2017 Parity Technologies (UK) Limited
Usage:
ethabi encode function <abi-path> <function-name> [-p <param>]... [-l | --lenient]
ethabi encode params [-v <type> <param>]... [-l | --lenient]
ethabi decode function <abi-path> <function-name> <data>
ethabi decode params [-t <type>]... <data>
ethabi decode log <abi-path> <event-name> [-l <topic>]... <data>
ethabi -h | --help
Options:
-h, --help Display this message and exit.
-l, --lenient Allow short representation of input params.
Commands:
encode Encode ABI call.
decode Decode ABI call result.
function Load function from json ABI file.
params Specify types of input params inline.
log Decode event log.
"#;

#[derive(Debug, RustcDecodable)]
struct Args {
cmd_encode: bool,
cmd_decode: bool,
cmd_function: bool,
cmd_params: bool,
cmd_log: bool,
arg_abi_path: String,
arg_function_name: String,
arg_event_name: String,
arg_param: Vec<String>,
arg_type: Vec<String>,
arg_data: String,
arg_topic: Vec<String>,
flag_lenient: bool,
}

fn main() {
let result = execute(env::args());

match result {
Ok(s) => println!("{}", s),
Err(error) => println!("error: {:?}", error)
}
}

fn execute<S, I>(command: I) -> Result<String, Error> where I: IntoIterator<Item=S>, S: AsRef<str> {
let args: Args = Docopt::new(ETHABI)
.and_then(|d| d.argv(command).decode())
.unwrap_or_else(|e| e.exit());

if args.cmd_encode && args.cmd_function {
encode_call(&args.arg_abi_path, args.arg_function_name, args.arg_param, args.flag_lenient)
} else if args.cmd_encode && args.cmd_params {
encode_params(args.arg_type, args.arg_param, args.flag_lenient)
} else if args.cmd_decode && args.cmd_function {
decode_call_output(&args.arg_abi_path, args.arg_function_name, args.arg_data)
} else if args.cmd_decode && args.cmd_params {
decode_params(args.arg_type, args.arg_data)
} else if args.cmd_decode && args.cmd_log {
decode_log(&args.arg_abi_path, args.arg_event_name, args.arg_topic, args.arg_data)
} else {
unreachable!()
}
}

fn load_function(path: &str, function: String) -> Result<Function, Error> {
let file = try!(File::open(path));
let bytes: Vec<u8> = try!(file.bytes().collect());

let interface = try!(Interface::load(&bytes));
let contract = Contract::new(interface);
let function = try!(contract.function(function));
Ok(function)
}

fn load_event(path: &str, event: String) -> Result<Event, Error> {
let file = try!(File::open(path));
let bytes: Vec<u8> = try!(file.bytes().collect());

let interface = try!(Interface::load(&bytes));
let contract = Contract::new(interface);
let event = try!(contract.event(event));
Ok(event)
}

fn parse_tokens(params: &[(ParamType, String)], lenient: bool) -> Result<Vec<Token>, Error> {
params.iter()
.map(|&(ref param, ref value)| match lenient {
true => LenientTokenizer::tokenize(param, value),
false => StrictTokenizer::tokenize(param, value)
})
.collect::<Result<_, _>>()
.map_err(From::from)
}

fn encode_call(path: &str, function: String, values: Vec<String>, lenient: bool) -> Result<String, Error> {
let function = try!(load_function(path, function));
let types = function.input_params();

let params: Vec<_> = types.into_iter()
.zip(values.into_iter())
.collect();

let tokens = try!(parse_tokens(&params, lenient));
let result = try!(function.encode_call(tokens));

Ok(result.to_hex())
}

fn encode_params(types: Vec<String>, values: Vec<String>, lenient: bool) -> Result<String, Error> {
assert_eq!(types.len(), values.len());

let types: Result<Vec<ParamType>, _> = types.iter()
.map(|s| Reader::read(s))
.collect();

let types = try!(types);

let params: Vec<_> = types.into_iter()
.zip(values.into_iter())
.collect();

let tokens = try!(parse_tokens(&params, lenient));
let result = Encoder::encode(tokens);

Ok(result.to_hex())
}

fn decode_call_output(path: &str, function: String, data: String) -> Result<String, Error> {
let function = try!(load_function(path, function));
let data = try!(data.from_hex());

let types = function.output_params();
let tokens = try!(function.decode_output(data));

assert_eq!(types.len(), tokens.len());

let result = types.iter()
.zip(tokens.iter())
.map(|(ty, to)| format!("{} {}", ty, to))
.collect::<Vec<String>>()
.join("\n");

Ok(result)
}

fn decode_params(types: Vec<String>, data: String) -> Result<String, Error> {
let types: Result<Vec<ParamType>, _> = types.iter()
.map(|s| Reader::read(s))
.collect();

let types = try!(types);
let data = try!(data.from_hex());

let tokens = try!(Decoder::decode(&types, data));

assert_eq!(types.len(), tokens.len());

let result = types.iter()
.zip(tokens.iter())
.map(|(ty, to)| format!("{} {}", ty, to))
.collect::<Vec<String>>()
.join("\n");

Ok(result)
}

fn decode_log(path: &str, event: String, topics: Vec<String>, data: String) -> Result<String, Error> {
let event = try!(load_event(path, event));
let topics: Result<Vec<[u8; 32]>, Error> = topics.into_iter()
.map(|t| t.token_from_hex().map_err(From::from))
.collect();
let topics = try!(topics);
let data = try!(data.from_hex());
let decoded = try!(event.decode_log(topics, data));

let result = decoded.params.into_iter()
.map(|(name, kind, value)| format!("{} {} {}", name, kind, value))
.collect::<Vec<String>>()
.join("\n");

Ok(result)
}

#[cfg(test)]
mod tests {
use super::execute;

#[test]
fn simple_encode() {
let command = "ethabi encode params -v bool 1".split(" ");
let expected = "0000000000000000000000000000000000000000000000000000000000000001";
assert_eq!(execute(command).unwrap(), expected);
}

// TODO: parsing negative values is not working
#[test]
#[ignore]
fn int_encode() {
let command = "ethabi encode paramas -v int256 -2 --lenient".split(" ");
let expected = "fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe";
assert_eq!(execute(command).unwrap(), expected);
}

#[test]
fn multi_encode() {
let command = "ethabi encode params -v bool 1 -v string gavofyork -v bool 0".split(" ");
let expected = "00000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000000000060000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000096761766f66796f726b0000000000000000000000000000000000000000000000";
assert_eq!(execute(command).unwrap(), expected);
}

#[test]
fn array_encode() {
let command = "ethabi encode params -v bool[] [1,0,false]".split(" ");
let expected = "00000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000000000003000000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000";
assert_eq!(execute(command).unwrap(), expected);
}

#[test]
fn abi_encode() {
let command = "ethabi encode function examples/test.json foo -p 1".split(" ");
let expected = "455575780000000000000000000000000000000000000000000000000000000000000001";
assert_eq!(execute(command).unwrap(), expected);
}

#[test]
fn simple_decode() {
let command = "ethabi decode params -t bool 0000000000000000000000000000000000000000000000000000000000000001".split(" ");
let expected = "bool true";
assert_eq!(execute(command).unwrap(), expected);
}

#[test]
fn int_decode() {
let command = "ethabi decode params -t int256 fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe".split(" ");
let expected = "int256 fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe";
assert_eq!(execute(command).unwrap(), expected);
}

#[test]
fn multi_decode() {
let command = "ethabi decode params -t bool -t string -t bool 00000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000000000060000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000096761766f66796f726b0000000000000000000000000000000000000000000000".split(" ");
let expected =
"bool true
string gavofyork
bool false";
assert_eq!(execute(command).unwrap(), expected);
}

#[test]
fn array_decode() {
let command = "ethabi decode params -t bool[] 00000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000000000003000000000000000000000000000000000000000000000000000000000000000100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000".split(" ");
let expected = "bool[] [true,false,false]";
assert_eq!(execute(command).unwrap(), expected);
}

#[test]
fn abi_decode() {
let command = "ethabi decode function ./examples/foo.json bar 0000000000000000000000000000000000000000000000000000000000000001".split(" ");
let expected = "bool true";
assert_eq!(execute(command).unwrap(), expected);
}

#[test]
fn log_decode() {
let command = "ethabi decode log ./examples/event.json Event -l 0000000000000000000000000000000000000000000000000000000000000001 0000000000000000000000004444444444444444444444444444444444444444".split(" ");
let expected =
"a bool true
b address 4444444444444444444444444444444444444444";
assert_eq!(execute(command).unwrap(), expected);
}
}
Loading

0 comments on commit 9ef6d92

Please sign in to comment.