Skip to content

Commit 9858602

Browse files
committed
support EVM-based chains: make support other major EVM-based chains, improve on commandline's argument handling
This changes uses clap crate to handle positional index to act as a required argument without us needing to involving handling with std::env::args ourselves. Now it supports BSC, Ethereum, and Polygon by using --chain (or -c) with possible values setup via clap crate (as well). It also helps in checking whether input values are valid. This project name has been changed from bsccontract-diff to contractdiff.
1 parent 0a127ad commit 9858602

File tree

6 files changed

+245
-64
lines changed

6 files changed

+245
-64
lines changed

Cargo.lock

+130-13
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Cargo.toml

+5-4
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,12 @@
11
[package]
2-
name = "bsccontract-diff"
3-
version = "0.1.0"
2+
name = "contractdiff"
3+
version = "1.0.0"
44
edition = "2021"
55
authors = ["Wasin Thonkaew <wasin@wasin.io>"]
66
description = "Commandline program to output color-diff of contract code between two contract addresses. Suitable to use against non-verified contract code for analysis."
7-
repository = "https://github.com/haxpor/bsccontract-diff"
7+
repository = "https://github.com/haxpor/contractdiff"
88
license = "MIT"
9-
keywords = ["bsc", "contract-address", "diff", "crypto", "blockchain"]
9+
keywords = ["contract", "contract-address", "diff", "crypto", "blockchain"]
1010
categories = ["command-line-utilities"]
1111
readme = "README.md"
1212

@@ -20,3 +20,4 @@ hex = "0.4.3"
2020
futures = "0.3.21"
2121
tokio = { version = "1.17.0", features = ["full"] }
2222
regex = "1.5.5"
23+
clap = { version = "3.1.15", features = ["derive"] }

README.md

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,11 @@
1-
# bsccontract-diff
1+
# contractdiff
22
Commandline program to output color-diff of contract code between two input
33
contract addresses. Suitable to use against non-verified contract code for analysis.
44

55
# Usage
66

77
```
8-
bsccontract-diff 0x00..... 0xab....
8+
contractdiff --chain <CHAIN> <ADDRESS1> <ADDRESS2>
99
```
1010

1111
wheres address needs to be prefixed with `0x`.
@@ -17,7 +17,7 @@ is malformed, or such address is not a contract address, but an EOA address.
1717

1818
Install it via `cargo` as follows
1919

20-
`cargo install bsccontract-diff`
20+
`cargo install contractdiff`
2121

2222
# License
2323
MIT, Wasin Thonkaew

src/main.rs

+34-44
Original file line numberDiff line numberDiff line change
@@ -1,60 +1,50 @@
1+
mod types;
2+
mod util;
3+
14
use similar::ChangeTag;
25
use console::Style;
6+
use clap::Parser;
37
use web3::types::Address;
4-
use regex::Regex;
5-
6-
use std::env;
7-
8-
/// Print usage text
9-
fn print_program_usage() {
10-
eprintln!("Usage: {} <contract-address-1> <contract-address-2>", env!("CARGO_PKG_NAME", "bsccontract-diff"));
11-
}
12-
13-
/// Check whether specified address string is an address.
14-
///
15-
/// This is not full-fledge checking in which it doesn't take into account
16-
/// checking of checksum address.
17-
///
18-
/// # Arguments
19-
/// * `address` - address string to check
20-
fn is_address_simplified(address: &str) -> bool {
21-
let lowercase_address = address.to_lowercase();
22-
let regex: Regex = Regex::new(r#"^(0x)?[0-9a-f]{40}$"#).unwrap();
23-
24-
regex.is_match(&lowercase_address)
25-
}
8+
use types::*;
9+
use util::*;
2610

2711
#[tokio::main]
2812
async fn main() {
29-
let args: Vec<String> = env::args().collect();
30-
31-
// check cli's arguments first
32-
// might need a better check, but for now it's sufficed.
33-
// Ignore arguments after required 2 addresses.
34-
if args.len() < 3 {
35-
print_program_usage();
36-
std::process::exit(1);
37-
}
13+
// required argument of first, and second one will be automatically handled
14+
// by clap as we set them as positional argument index.
15+
let cmd_args = CommandlineArgs::parse();
3816

3917
// pre-check of address arguments
40-
if !is_address_simplified(&args[1]) {
18+
if !is_address_simplified(&cmd_args.address1) {
4119
eprintln!("Error: 1st address is malformed. Make sure to prefix with '0x' and has 40 characters in length (exclude `0x`).");
4220
std::process::exit(1);
4321
}
44-
if !is_address_simplified(&args[2]) {
22+
if !is_address_simplified(&cmd_args.address2) {
4523
eprintln!("Error: 2nd address is malformed. Make sure to prefix with '0x' and has 40 characters in length (exclude `0x`).");
4624
std::process::exit(1);
4725
}
4826

49-
// create web3 related context
50-
let http = match web3::transports::Http::new("https://bsc-dataseed.binance.org/") {
51-
Ok(res) => res,
52-
Err(e) => {
53-
eprintln!("Error: creating web3's http; err={}", e);
54-
std::process::exit(1);
55-
}
56-
};
57-
let web3 = web3::Web3::new(http);
27+
// validate value of chain flag option
28+
let chain_value = cmd_args.chain.to_lowercase();
29+
let mut chain: Option<ChainType> = None;
30+
if chain_value == "bsc" {
31+
chain = Some(ChainType::BSC);
32+
}
33+
else if chain_value == "ethereum" {
34+
chain = Some(ChainType::Ethereum);
35+
}
36+
else if chain_value == "polygon" {
37+
chain = Some(ChainType::Polygon);
38+
}
39+
// NOTE: no need for else case here as clap crate handles non-valid values
40+
// for us.
41+
42+
if chain.is_none() {
43+
eprintln!("Error unexpected thing happen affecting us not to determine which chain to operate on");
44+
std::process::exit(1);
45+
}
46+
47+
let web3 = create_web3(chain.unwrap());
5848

5949
// CAVEAT: we cannot do early check whether the input address is indeed
6050
// a contract address, but until we get response of bytecode of back.
@@ -63,7 +53,7 @@ async fn main() {
6353
// If it is EOA address -> empty (0-length string)
6454

6555
// get bytecode from specified contract address of both arguments
66-
let contract1_hexbytes_decoded = match hex::decode(&args[1][2..]) {
56+
let contract1_hexbytes_decoded = match hex::decode(&cmd_args.address1[2..]) {
6757
Ok(res) => res,
6858
Err(e) => {
6959
eprintln!("Error: hex decoding of 1st address; err={}", e);
@@ -73,7 +63,7 @@ async fn main() {
7363
let contract1_code_fut = web3.eth().code(Address::from_slice(contract1_hexbytes_decoded.as_slice()), None);
7464

7565
// do the same for 2nd contract address
76-
let contract2_hexbytes_decoded = match hex::decode(&args[2][2..]) {
66+
let contract2_hexbytes_decoded = match hex::decode(&cmd_args.address2[2..]) {
7767
Ok(res) => res,
7868
Err(e) => {
7969
eprintln!("Error: hex decoding of 2nd address; err={}", e);

0 commit comments

Comments
 (0)