Skip to content
This repository has been archived by the owner on Oct 19, 2024. It is now read-only.

Commit

Permalink
fix(eip712): improve argument parsing (#1485)
Browse files Browse the repository at this point in the history
* fix(eip712): improve argument parsing

* use reexported serde_json

* chore: rustfmt

* fix: add missing salt
  • Loading branch information
mattsse authored Jul 20, 2022
1 parent 1d4f3ef commit a958b4f
Show file tree
Hide file tree
Showing 4 changed files with 49 additions and 35 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -42,7 +42,7 @@ pub(crate) fn struct_declaration(cx: &Context) -> TokenStream {

let abi_parse = if !cx.human_readable {
quote! {
pub static #abi_name: #ethers_contract::Lazy<#ethers_core::abi::Abi> = #ethers_contract::Lazy::new(|| serde_json::from_str(#abi)
pub static #abi_name: #ethers_contract::Lazy<#ethers_core::abi::Abi> = #ethers_contract::Lazy::new(|| #ethers_core::utils::__serde_json::from_str(#abi)
.expect("invalid abi"));
}
} else {
Expand Down
7 changes: 3 additions & 4 deletions ethers-core/ethers-derive-eip712/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,6 @@
//! determine if there is a nested eip712 struct. However, this work is not yet complete.
#![deny(rustdoc::broken_intra_doc_links)]
#![deny(unused_crate_dependencies)]

use std::convert::TryFrom;

Expand Down Expand Up @@ -121,21 +120,21 @@ fn impl_eip_712_macro(ast: &syn::DeriveInput) -> TokenStream {

fn type_hash() -> Result<[u8; 32], Self::Error> {
use std::convert::TryFrom;
let decoded = hex::decode(#type_hash)?;
let decoded = #ethers_core::utils::hex::decode(#type_hash)?;
let byte_array: [u8; 32] = <[u8; 32]>::try_from(&decoded[..])?;
Ok(byte_array)
}

// Return the pre-computed domain separator from compile time;
fn domain_separator(&self) -> Result<[u8; 32], Self::Error> {
use std::convert::TryFrom;
let decoded = hex::decode(#domain_separator)?;
let decoded = #ethers_core::utils::hex::decode(#domain_separator)?;
let byte_array: [u8; 32] = <[u8; 32]>::try_from(&decoded[..])?;
Ok(byte_array)
}

fn domain(&self) -> Result<#ethers_core::types::transaction::eip712::EIP712Domain, Self::Error> {
let domain: #ethers_core::types::transaction::eip712::EIP712Domain = serde_json::from_str(#domain_str)?;
let domain: #ethers_core::types::transaction::eip712::EIP712Domain = # ethers_core::utils::__serde_json::from_str(#domain_str)?;

Ok(domain)
}
Expand Down
69 changes: 39 additions & 30 deletions ethers-core/src/types/transaction/eip712.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ use syn::{
use crate::{
abi,
abi::{ParamType, Token},
types::{Address, H160, U256},
types::{Address, U256},
utils::keccak256,
};

Expand Down Expand Up @@ -196,9 +196,14 @@ impl TryFrom<&syn::DeriveInput> for EIP712Domain {
type Error = TokenStream;
fn try_from(input: &syn::DeriveInput) -> Result<EIP712Domain, Self::Error> {
let mut domain = EIP712Domain::default();
let mut domain_name = None;
let mut domain_version = None;
let mut chain_id = None;
let mut verifying_contract = None;

let mut found_eip712_attribute = false;

for attribute in input.attrs.iter() {
'attribute_search: for attribute in input.attrs.iter() {
if let AttrStyle::Outer = attribute.style {
if let Ok(syn::Meta::List(meta)) = attribute.parse_meta() {
if meta.path.is_ident("eip712") {
Expand All @@ -219,15 +224,15 @@ impl TryFrom<&syn::DeriveInput> for EIP712Domain {
match ident.to_string().as_ref() {
"name" => match meta.lit {
syn::Lit::Str(ref lit_str) => {
if domain.name != String::default() {
if domain_name.is_some() {
return Err(Error::new(
meta.path.span(),
"domain name already specified",
)
.to_compile_error())
}

domain.name = lit_str.value();
domain_name = Some(lit_str.value());
}
_ => {
return Err(Error::new(
Expand All @@ -239,15 +244,15 @@ impl TryFrom<&syn::DeriveInput> for EIP712Domain {
},
"version" => match meta.lit {
syn::Lit::Str(ref lit_str) => {
if domain.version != String::default() {
if domain_version.is_some() {
return Err(Error::new(
meta.path.span(),
"domain version already specified",
)
.to_compile_error())
}

domain.version = lit_str.value();
domain_version = Some(lit_str.value());
}
_ => {
return Err(Error::new(
Expand All @@ -259,15 +264,15 @@ impl TryFrom<&syn::DeriveInput> for EIP712Domain {
},
"chain_id" => match meta.lit {
syn::Lit::Int(ref lit_int) => {
if domain.chain_id != U256::default() {
if chain_id.is_some() {
return Err(Error::new(
meta.path.span(),
"domain chain_id already specified",
)
.to_compile_error())
}

domain.chain_id = U256::from(
chain_id = Some(U256::from(
lit_int.base10_parse::<u64>().map_err(
|_| {
Error::new(
Expand All @@ -277,7 +282,7 @@ impl TryFrom<&syn::DeriveInput> for EIP712Domain {
.to_compile_error()
},
)?,
);
));
}
_ => {
return Err(Error::new(
Expand All @@ -289,22 +294,21 @@ impl TryFrom<&syn::DeriveInput> for EIP712Domain {
},
"verifying_contract" => match meta.lit {
syn::Lit::Str(ref lit_str) => {
if domain.verifying_contract != H160::default()
{
if verifying_contract.is_some() {
return Err(Error::new(
meta.path.span(),
"domain verifying_contract already specified",
)
.to_compile_error());
}

domain.verifying_contract = lit_str.value().parse().map_err(|_| {
verifying_contract = Some(lit_str.value().parse().map_err(|_| {
Error::new(
meta.path.span(),
"failed to parse verifying contract into Address",
)
.to_compile_error()
})?;
})?);
}
_ => {
return Err(Error::new(
Expand All @@ -316,7 +320,7 @@ impl TryFrom<&syn::DeriveInput> for EIP712Domain {
},
"salt" => match meta.lit {
syn::Lit::Str(ref lit_str) => {
if domain.salt != Option::None {
if domain.salt.is_some() {
return Err(Error::new(
meta.path.span(),
"domain salt already specified",
Expand Down Expand Up @@ -365,36 +369,41 @@ impl TryFrom<&syn::DeriveInput> for EIP712Domain {
}
}

if domain.name == String::default() {
return Err(Error::new(
domain.name = domain_name.ok_or_else(|| {
Error::new(
meta.path.span(),
"missing required domain attribute: 'name'".to_string(),
)
.to_compile_error())
}
if domain.version == String::default() {
return Err(Error::new(
.to_compile_error()
})?;

domain.version = domain_version.ok_or_else(|| {
Error::new(
meta.path.span(),
"missing required domain attribute: 'version'".to_string(),
)
.to_compile_error())
}
if domain.chain_id == U256::default() {
return Err(Error::new(
.to_compile_error()
})?;

domain.chain_id = chain_id.ok_or_else(|| {
Error::new(
meta.path.span(),
"missing required domain attribute: 'chain_id'".to_string(),
)
.to_compile_error())
}
if domain.verifying_contract == H160::default() {
return Err(Error::new(
.to_compile_error()
})?;

domain.verifying_contract = verifying_contract.ok_or_else(|| {
Error::new(
meta.path.span(),
"missing required domain attribute: 'verifying_contract'"
.to_string(),
)
.to_compile_error())
}
.to_compile_error()
})?;
}

break 'attribute_search
}
}
}
Expand Down
6 changes: 6 additions & 0 deletions ethers-core/src/utils/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,12 @@ use k256::{ecdsa::SigningKey, PublicKey as K256PublicKey};
use std::convert::{TryFrom, TryInto};
use thiserror::Error;

/// Re-export of serde-json
#[doc(hidden)]
pub mod __serde_json {
pub use serde_json::*;
}

#[derive(Error, Debug)]
pub enum ConversionError {
#[error("Unknown units: {0}")]
Expand Down

0 comments on commit a958b4f

Please sign in to comment.