From 121e3ecd31276faffe98fe0ef5a7180c0f223191 Mon Sep 17 00:00:00 2001
From: danielw1534 <39487081+danielw1534@users.noreply.github.com>
Date: Wed, 24 Jul 2024 12:08:08 -0500
Subject: [PATCH] Fix for deployed smart wallet signatures.
Co-authored-by: danielw1534 <39487081+danielw1534@users.noreply.github.com>
---
Cargo.lock | 2 +-
Cargo.toml | 5 +-
build.rs | 1 +
contracts/DeployedSmartWalletAbi.json | 758 ++++++++++++++++++++++++++
contracts/Erc1271Mock.sol | 2 +-
src/lib.rs | 343 ++++++++----
src/test_helpers.rs | 19 +-
7 files changed, 1012 insertions(+), 118 deletions(-)
create mode 100644 contracts/DeployedSmartWalletAbi.json
diff --git a/Cargo.lock b/Cargo.lock
index d69d2c0..27467aa 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -1478,7 +1478,7 @@ dependencies = [
[[package]]
name = "eth-signature-verifier"
-version = "0.1.0"
+version = "0.2.0"
dependencies = [
"alloy",
"alloy-node-bindings",
diff --git a/Cargo.toml b/Cargo.toml
index 33139f2..3a4f394 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -1,7 +1,7 @@
[package]
name = "eth-signature-verifier"
description = "Universal Etheruem signature verification with ERC-6492"
-version = "0.1.0"
+version = "0.2.0"
edition = "2021"
authors = ["Royal Engineering"]
license = "MIT"
@@ -18,8 +18,9 @@ alloy = { version = "0.2", features = [
"rpc-types",
"transports",
"signers",
+ "transport-http",
"signer-local",
- "sol-types"
+ "sol-types",
]}
log = "0.4"
thiserror = "1.0.63"
diff --git a/build.rs b/build.rs
index 6723015..4feb335 100644
--- a/build.rs
+++ b/build.rs
@@ -77,6 +77,7 @@ const ERC6492_FILE: &str = "forge/out/Erc6492.sol/ValidateSigOffchain.json";
const ERC6492_BYTECODE_FILE: &str = "forge/out/Erc6492.sol/ValidateSigOffchain.bytecode";
const ERC1271_MOCK_FILE: &str = "forge/out/Erc1271Mock.sol/Erc1271Mock.json";
const ERC1271_MOCK_BYTECODE_FILE: &str = "forge/out/Erc1271Mock.sol/Erc1271Mock.bytecode";
+
fn extract_bytecodes() {
extract_bytecode(
&format_foundry_dir(ERC6492_FILE),
diff --git a/contracts/DeployedSmartWalletAbi.json b/contracts/DeployedSmartWalletAbi.json
new file mode 100644
index 0000000..4d351ab
--- /dev/null
+++ b/contracts/DeployedSmartWalletAbi.json
@@ -0,0 +1,758 @@
+[{
+ "inputs": [],
+ "stateMutability": "nonpayable",
+ "type": "constructor"
+ }, {
+ "inputs": [
+ {
+ "internalType": "bytes",
+ "name": "owner",
+ "type": "bytes"
+ }
+ ],
+ "name": "AlreadyOwner",
+ "type": "error"
+ }, {
+ "inputs": [],
+ "name": "Initialized",
+ "type": "error"
+ }, {
+ "inputs": [
+ {
+ "internalType": "bytes",
+ "name": "owner",
+ "type": "bytes"
+ }
+ ],
+ "name": "InvalidEthereumAddressOwner",
+ "type": "error"
+ }, {
+ "inputs": [
+ {
+ "internalType": "uint256",
+ "name": "key",
+ "type": "uint256"
+ }
+ ],
+ "name": "InvalidNonceKey",
+ "type": "error"
+ }, {
+ "inputs": [
+ {
+ "internalType": "bytes",
+ "name": "owner",
+ "type": "bytes"
+ }
+ ],
+ "name": "InvalidOwnerBytesLength",
+ "type": "error"
+ }, {
+ "inputs": [],
+ "name": "LastOwner",
+ "type": "error"
+ }, {
+ "inputs": [
+ {
+ "internalType": "uint256",
+ "name": "index",
+ "type": "uint256"
+ }
+ ],
+ "name": "NoOwnerAtIndex",
+ "type": "error"
+ }, {
+ "inputs": [
+ {
+ "internalType": "uint256",
+ "name": "ownersRemaining",
+ "type": "uint256"
+ }
+ ],
+ "name": "NotLastOwner",
+ "type": "error"
+ }, {
+ "inputs": [
+ {
+ "internalType": "bytes4",
+ "name": "selector",
+ "type": "bytes4"
+ }
+ ],
+ "name": "SelectorNotAllowed",
+ "type": "error"
+ }, {
+ "inputs": [],
+ "name": "Unauthorized",
+ "type": "error"
+ }, {
+ "inputs": [],
+ "name": "UnauthorizedCallContext",
+ "type": "error"
+ }, {
+ "inputs": [],
+ "name": "UpgradeFailed",
+ "type": "error"
+ }, {
+ "inputs": [
+ {
+ "internalType": "uint256",
+ "name": "index",
+ "type": "uint256"
+ },
+ {
+ "internalType": "bytes",
+ "name": "expectedOwner",
+ "type": "bytes"
+ },
+ {
+ "internalType": "bytes",
+ "name": "actualOwner",
+ "type": "bytes"
+ }
+ ],
+ "name": "WrongOwnerAtIndex",
+ "type": "error"
+ }, {
+ "anonymous": false,
+ "inputs": [
+ {
+ "indexed": true,
+ "internalType": "uint256",
+ "name": "index",
+ "type": "uint256"
+ },
+ {
+ "indexed": false,
+ "internalType": "bytes",
+ "name": "owner",
+ "type": "bytes"
+ }
+ ],
+ "name": "AddOwner",
+ "type": "event"
+ }, {
+ "anonymous": false,
+ "inputs": [
+ {
+ "indexed": true,
+ "internalType": "uint256",
+ "name": "index",
+ "type": "uint256"
+ },
+ {
+ "indexed": false,
+ "internalType": "bytes",
+ "name": "owner",
+ "type": "bytes"
+ }
+ ],
+ "name": "RemoveOwner",
+ "type": "event"
+ }, {
+ "anonymous": false,
+ "inputs": [
+ {
+ "indexed": true,
+ "internalType": "address",
+ "name": "implementation",
+ "type": "address"
+ }
+ ],
+ "name": "Upgraded",
+ "type": "event"
+ }, {
+ "stateMutability": "payable",
+ "type": "fallback"
+ }, {
+ "inputs": [],
+ "name": "REPLAYABLE_NONCE_KEY",
+ "outputs": [
+ {
+ "internalType": "uint256",
+ "name": "",
+ "type": "uint256"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "address",
+ "name": "owner",
+ "type": "address"
+ }
+ ],
+ "name": "addOwnerAddress",
+ "outputs": [],
+ "stateMutability": "nonpayable",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "bytes32",
+ "name": "x",
+ "type": "bytes32"
+ },
+ {
+ "internalType": "bytes32",
+ "name": "y",
+ "type": "bytes32"
+ }
+ ],
+ "name": "addOwnerPublicKey",
+ "outputs": [],
+ "stateMutability": "nonpayable",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "bytes4",
+ "name": "functionSelector",
+ "type": "bytes4"
+ }
+ ],
+ "name": "canSkipChainIdValidation",
+ "outputs": [
+ {
+ "internalType": "bool",
+ "name": "",
+ "type": "bool"
+ }
+ ],
+ "stateMutability": "pure",
+ "type": "function"
+ }, {
+ "inputs": [],
+ "name": "domainSeparator",
+ "outputs": [
+ {
+ "internalType": "bytes32",
+ "name": "",
+ "type": "bytes32"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [],
+ "name": "eip712Domain",
+ "outputs": [
+ {
+ "internalType": "bytes1",
+ "name": "fields",
+ "type": "bytes1"
+ },
+ {
+ "internalType": "string",
+ "name": "name",
+ "type": "string"
+ },
+ {
+ "internalType": "string",
+ "name": "version",
+ "type": "string"
+ },
+ {
+ "internalType": "uint256",
+ "name": "chainId",
+ "type": "uint256"
+ },
+ {
+ "internalType": "address",
+ "name": "verifyingContract",
+ "type": "address"
+ },
+ {
+ "internalType": "bytes32",
+ "name": "salt",
+ "type": "bytes32"
+ },
+ {
+ "internalType": "uint256[]",
+ "name": "extensions",
+ "type": "uint256[]"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [],
+ "name": "entryPoint",
+ "outputs": [
+ {
+ "internalType": "address",
+ "name": "",
+ "type": "address"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "address",
+ "name": "target",
+ "type": "address"
+ },
+ {
+ "internalType": "uint256",
+ "name": "value",
+ "type": "uint256"
+ },
+ {
+ "internalType": "bytes",
+ "name": "data",
+ "type": "bytes"
+ }
+ ],
+ "name": "execute",
+ "outputs": [],
+ "stateMutability": "payable",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "components": [
+ {
+ "internalType": "address",
+ "name": "target",
+ "type": "address"
+ },
+ {
+ "internalType": "uint256",
+ "name": "value",
+ "type": "uint256"
+ },
+ {
+ "internalType": "bytes",
+ "name": "data",
+ "type": "bytes"
+ }
+ ],
+ "internalType": "struct CoinbaseSmartWallet.Call[]",
+ "name": "calls",
+ "type": "tuple[]"
+ }
+ ],
+ "name": "executeBatch",
+ "outputs": [],
+ "stateMutability": "payable",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "bytes[]",
+ "name": "calls",
+ "type": "bytes[]"
+ }
+ ],
+ "name": "executeWithoutChainIdValidation",
+ "outputs": [],
+ "stateMutability": "payable",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "components": [
+ {
+ "internalType": "address",
+ "name": "sender",
+ "type": "address"
+ },
+ {
+ "internalType": "uint256",
+ "name": "nonce",
+ "type": "uint256"
+ },
+ {
+ "internalType": "bytes",
+ "name": "initCode",
+ "type": "bytes"
+ },
+ {
+ "internalType": "bytes",
+ "name": "callData",
+ "type": "bytes"
+ },
+ {
+ "internalType": "uint256",
+ "name": "callGasLimit",
+ "type": "uint256"
+ },
+ {
+ "internalType": "uint256",
+ "name": "verificationGasLimit",
+ "type": "uint256"
+ },
+ {
+ "internalType": "uint256",
+ "name": "preVerificationGas",
+ "type": "uint256"
+ },
+ {
+ "internalType": "uint256",
+ "name": "maxFeePerGas",
+ "type": "uint256"
+ },
+ {
+ "internalType": "uint256",
+ "name": "maxPriorityFeePerGas",
+ "type": "uint256"
+ },
+ {
+ "internalType": "bytes",
+ "name": "paymasterAndData",
+ "type": "bytes"
+ },
+ {
+ "internalType": "bytes",
+ "name": "signature",
+ "type": "bytes"
+ }
+ ],
+ "internalType": "struct UserOperation",
+ "name": "userOp",
+ "type": "tuple"
+ }
+ ],
+ "name": "getUserOpHashWithoutChainId",
+ "outputs": [
+ {
+ "internalType": "bytes32",
+ "name": "",
+ "type": "bytes32"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [],
+ "name": "implementation",
+ "outputs": [
+ {
+ "internalType": "address",
+ "name": "$",
+ "type": "address"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "bytes[]",
+ "name": "owners",
+ "type": "bytes[]"
+ }
+ ],
+ "name": "initialize",
+ "outputs": [],
+ "stateMutability": "payable",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "address",
+ "name": "account",
+ "type": "address"
+ }
+ ],
+ "name": "isOwnerAddress",
+ "outputs": [
+ {
+ "internalType": "bool",
+ "name": "",
+ "type": "bool"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "bytes",
+ "name": "account",
+ "type": "bytes"
+ }
+ ],
+ "name": "isOwnerBytes",
+ "outputs": [
+ {
+ "internalType": "bool",
+ "name": "",
+ "type": "bool"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "bytes32",
+ "name": "x",
+ "type": "bytes32"
+ },
+ {
+ "internalType": "bytes32",
+ "name": "y",
+ "type": "bytes32"
+ }
+ ],
+ "name": "isOwnerPublicKey",
+ "outputs": [
+ {
+ "internalType": "bool",
+ "name": "",
+ "type": "bool"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "bytes32",
+ "name": "hash",
+ "type": "bytes32"
+ },
+ {
+ "internalType": "bytes",
+ "name": "signature",
+ "type": "bytes"
+ }
+ ],
+ "name": "isValidSignature",
+ "outputs": [
+ {
+ "internalType": "bytes4",
+ "name": "result",
+ "type": "bytes4"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [],
+ "name": "nextOwnerIndex",
+ "outputs": [
+ {
+ "internalType": "uint256",
+ "name": "",
+ "type": "uint256"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "uint256",
+ "name": "index",
+ "type": "uint256"
+ }
+ ],
+ "name": "ownerAtIndex",
+ "outputs": [
+ {
+ "internalType": "bytes",
+ "name": "",
+ "type": "bytes"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [],
+ "name": "ownerCount",
+ "outputs": [
+ {
+ "internalType": "uint256",
+ "name": "",
+ "type": "uint256"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [],
+ "name": "proxiableUUID",
+ "outputs": [
+ {
+ "internalType": "bytes32",
+ "name": "",
+ "type": "bytes32"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "uint256",
+ "name": "index",
+ "type": "uint256"
+ },
+ {
+ "internalType": "bytes",
+ "name": "owner",
+ "type": "bytes"
+ }
+ ],
+ "name": "removeLastOwner",
+ "outputs": [],
+ "stateMutability": "nonpayable",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "uint256",
+ "name": "index",
+ "type": "uint256"
+ },
+ {
+ "internalType": "bytes",
+ "name": "owner",
+ "type": "bytes"
+ }
+ ],
+ "name": "removeOwnerAtIndex",
+ "outputs": [],
+ "stateMutability": "nonpayable",
+ "type": "function"
+ }, {
+ "inputs": [],
+ "name": "removedOwnersCount",
+ "outputs": [
+ {
+ "internalType": "uint256",
+ "name": "",
+ "type": "uint256"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "bytes32",
+ "name": "hash",
+ "type": "bytes32"
+ }
+ ],
+ "name": "replaySafeHash",
+ "outputs": [
+ {
+ "internalType": "bytes32",
+ "name": "",
+ "type": "bytes32"
+ }
+ ],
+ "stateMutability": "view",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "internalType": "address",
+ "name": "newImplementation",
+ "type": "address"
+ },
+ {
+ "internalType": "bytes",
+ "name": "data",
+ "type": "bytes"
+ }
+ ],
+ "name": "upgradeToAndCall",
+ "outputs": [],
+ "stateMutability": "payable",
+ "type": "function"
+ }, {
+ "inputs": [
+ {
+ "components": [
+ {
+ "internalType": "address",
+ "name": "sender",
+ "type": "address"
+ },
+ {
+ "internalType": "uint256",
+ "name": "nonce",
+ "type": "uint256"
+ },
+ {
+ "internalType": "bytes",
+ "name": "initCode",
+ "type": "bytes"
+ },
+ {
+ "internalType": "bytes",
+ "name": "callData",
+ "type": "bytes"
+ },
+ {
+ "internalType": "uint256",
+ "name": "callGasLimit",
+ "type": "uint256"
+ },
+ {
+ "internalType": "uint256",
+ "name": "verificationGasLimit",
+ "type": "uint256"
+ },
+ {
+ "internalType": "uint256",
+ "name": "preVerificationGas",
+ "type": "uint256"
+ },
+ {
+ "internalType": "uint256",
+ "name": "maxFeePerGas",
+ "type": "uint256"
+ },
+ {
+ "internalType": "uint256",
+ "name": "maxPriorityFeePerGas",
+ "type": "uint256"
+ },
+ {
+ "internalType": "bytes",
+ "name": "paymasterAndData",
+ "type": "bytes"
+ },
+ {
+ "internalType": "bytes",
+ "name": "signature",
+ "type": "bytes"
+ }
+ ],
+ "internalType": "struct UserOperation",
+ "name": "userOp",
+ "type": "tuple"
+ },
+ {
+ "internalType": "bytes32",
+ "name": "userOpHash",
+ "type": "bytes32"
+ },
+ {
+ "internalType": "uint256",
+ "name": "missingAccountFunds",
+ "type": "uint256"
+ }
+ ],
+ "name": "validateUserOp",
+ "outputs": [
+ {
+ "internalType": "uint256",
+ "name": "validationData",
+ "type": "uint256"
+ }
+ ],
+ "stateMutability": "nonpayable",
+ "type": "function"
+ }, {
+ "stateMutability": "payable",
+ "type": "receive"
+ }]
\ No newline at end of file
diff --git a/contracts/Erc1271Mock.sol b/contracts/Erc1271Mock.sol
index a79f0dc..b8cc90b 100644
--- a/contracts/Erc1271Mock.sol
+++ b/contracts/Erc1271Mock.sol
@@ -78,4 +78,4 @@ contract Erc1271Mock {
return signer;
}
-}
+}
\ No newline at end of file
diff --git a/src/lib.rs b/src/lib.rs
index 033220e..2e90314 100644
--- a/src/lib.rs
+++ b/src/lib.rs
@@ -5,7 +5,7 @@ use {
rpc::types::{TransactionInput, TransactionRequest},
signers::k256::ecdsa::{RecoveryId, Signature, VerifyingKey},
sol,
- sol_types::SolConstructor,
+ sol_types::{SolCall, SolConstructor},
transports::{Transport, TransportErrorKind},
},
log::error,
@@ -14,7 +14,6 @@ use {
/// The expected result for a successful signature verification.
const SUCCESS_RESULT: u8 = 0x01;
-/// The magic bytes used to detect ERC-6492 signatures.
const ERC6492_DETECTION_SUFFIX: [u8; 32] = [
0x64, 0x92, 0x64, 0x92, 0x64, 0x92, 0x64, 0x92, 0x64, 0x92, 0x64, 0x92, 0x64, 0x92, 0x64, 0x92,
0x64, 0x92, 0x64, 0x92, 0x64, 0x92, 0x64, 0x92, 0x64, 0x92, 0x64, 0x92, 0x64, 0x92, 0x64, 0x92,
@@ -25,6 +24,11 @@ const VALIDATE_SIG_OFFCHAIN_BYTECODE: &[u8] = include_bytes!(concat!(
"/../../../../.foundry/forge/out/Erc6492.sol/ValidateSigOffchain.bytecode"
));
+const ERC1271_MOCK_BYTECODE: &[u8] = include_bytes!(concat!(
+ env!("OUT_DIR"),
+ "/../../../../.foundry/forge/out/Erc1271Mock.sol/Erc1271Mock.bytecode"
+));
+
sol! {
contract ValidateSigOffchain {
constructor (address _signer, bytes32 _hash, bytes memory _signature);
@@ -42,6 +46,11 @@ sol! {
contract Create2 {
function computeAddress(bytes32 salt, bytes32 bytecodeHash) external view returns (address) {}
}
+
+ contract Erc1271Mock {
+ function isValidSignature(bytes32 _hash, bytes calldata _signature) external view returns (bytes4) {}
+ }
+
}
sol! {
@@ -171,6 +180,7 @@ where
if signature.len() < 20 {
return Err(SignatureError::InvalidSignature);
}
+
Ok(Address::from_slice(&signature[0..20]))
}
}
@@ -201,21 +211,69 @@ fn ecrecover_address(message: B256, signature: &[u8]) -> Result
(
+pub async fn verify_signature_auto(
signature: S,
message: FixedBytes<32>,
provider: P,
) -> Result
where
S: Into + Clone,
- // M: AsRef<[u8]> + Clone,
P: Provider,
T: Transport + Clone,
{
let address = extract_address(signature.clone(), message, &provider)
.await
.unwrap();
- verify_signature(signature, address, message, provider).await
+
+ let code_at_address = provider
+ .get_code_at(address)
+ .await
+ .map_err(SignatureError::ProviderError)
+ .unwrap();
+
+ if code_at_address.is_empty() {
+ verify_signature(signature, address, message, provider).await
+ } else {
+ let call = Erc1271Mock::isValidSignatureCall {
+ _hash: message,
+ _signature: signature.into(),
+ };
+ let bytes = ERC1271_MOCK_BYTECODE
+ .iter()
+ .cloned()
+ .chain(call.abi_encode())
+ .collect::>();
+
+ let transaction_request =
+ TransactionRequest::default().input(TransactionInput::new(bytes.into()));
+
+ let result = provider.call(&transaction_request).await;
+
+ match result {
+ Err(e) => {
+ if let Some(error_response) = e.as_error_resp() {
+ if error_response.message.starts_with("execution reverted") {
+ Ok(Verification::Invalid)
+ } else {
+ Err(e)
+ }
+ } else {
+ Err(e)
+ }
+ }
+ Ok(result) => {
+ if let Some(result) = result.first() {
+ if result == &SUCCESS_RESULT {
+ Ok(Verification::Valid)
+ } else {
+ Ok(Verification::Invalid)
+ }
+ } else {
+ Ok(Verification::Invalid)
+ }
+ }
+ }
+ }
}
/// Recovers the signer's address from an ECDSA signature.
@@ -252,6 +310,18 @@ fn ecrecover(hash: B256, v: u8, r: U256, s: U256) -> Result,
T: Transport + Clone,
{
- let call = ValidateSigOffchain::constructorCall {
- _signer: address,
- _hash: message,
- _signature: signature.into(),
- };
- let bytes = VALIDATE_SIG_OFFCHAIN_BYTECODE
- .iter()
- .cloned()
- .chain(call.abi_encode())
- .collect::>();
- let transaction_request =
- TransactionRequest::default().input(TransactionInput::new(bytes.into()));
-
- let result = provider.call(&transaction_request).await;
-
- match result {
- Err(e) => {
- if let Some(error_response) = e.as_error_resp() {
- if error_response.message.starts_with("execution reverted") {
- Ok(Verification::Invalid)
- } else {
- Err(e)
+ let code_exists = provider
+ .get_code_at(address)
+ .await
+ .map_err(SignatureError::ProviderError)
+ .map(|code| !code.is_empty())
+ .unwrap();
+ if code_exists {
+ let call_request = TransactionRequest::default()
+ .to(address)
+ .input(TransactionInput::new(
+ Erc1271Mock::isValidSignatureCall {
+ _hash: message,
+ _signature: signature.into(),
}
+ .abi_encode()
+ .into(),
+ ));
+
+ let result = provider.call(&call_request).await;
+
+ let result = result.unwrap();
+ let magic = result.get(..4);
+ if let Some(magic) = magic {
+ if magic == MAGIC_VALUE.to_be_bytes().to_vec() {
+ Ok(Verification::Valid)
} else {
- Err(e)
+ Ok(Verification::Invalid)
}
+ } else {
+ Ok(Verification::Invalid)
}
- Ok(result) => {
- if let Some(result) = result.first() {
- if result == &SUCCESS_RESULT {
- Ok(Verification::Valid)
+ } else {
+ let call = ValidateSigOffchain::constructorCall {
+ _signer: address,
+ _hash: message,
+ _signature: signature.into(),
+ };
+ let bytes = VALIDATE_SIG_OFFCHAIN_BYTECODE
+ .iter()
+ .cloned()
+ .chain(call.abi_encode())
+ .collect::>();
+ let transaction_request =
+ TransactionRequest::default().input(TransactionInput::new(bytes.into()));
+
+ let result = provider.call(&transaction_request).await;
+ match result {
+ Err(e) => {
+ if let Some(error_response) = e.as_error_resp() {
+ if error_response.message.starts_with("execution reverted") {
+ Ok(Verification::Invalid)
+ } else {
+ Err(e)
+ }
+ } else {
+ Err(e)
+ }
+ }
+ Ok(result) => {
+ if let Some(result) = result.first() {
+ if result == &SUCCESS_RESULT {
+ Ok(Verification::Valid)
+ } else {
+ Ok(Verification::Invalid)
+ }
} else {
Ok(Verification::Invalid)
}
- } else {
- Ok(Verification::Invalid)
}
}
}
@@ -326,8 +428,11 @@ mod test {
super::*,
alloy::{
dyn_abi::TypedData,
- primitives::{address, b256, bytes, keccak256, Address, Uint, B256, I256, U256},
- providers::{network::Ethereum, ReqwestProvider},
+ primitives::{
+ address, b256, bytes, eip191_hash_message, keccak256, Address, Uint, B256, I256,
+ U256,
+ },
+ providers::{network::Ethereum, Provider, ReqwestProvider},
signers::{k256::ecdsa::SigningKey, local::PrivateKeySigner, Signer, SignerSync},
sol,
sol_types::{eip712_domain, SolCall, SolValue},
@@ -339,32 +444,9 @@ mod test {
},
};
- // Manual test. Paste address, signature, message, and project ID to verify
- // function
- #[tokio::test]
- #[ignore]
- async fn manual() {
- let address = address!("AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA");
- let message = "xxx";
- let signature = bytes!("aaaa");
-
- let provider = ReqwestProvider::::new_http(
- "https://rpc.walletconnect.com/v1?chainId=eip155:1&projectId=xxx"
- .parse()
- .unwrap(),
- );
- let message_bytes = message_str_to_bytes(message);
- assert!(
- verify_signature(signature, address, message_bytes, provider)
- .await
- .unwrap()
- .is_valid()
- );
- }
-
#[tokio::test]
async fn test_extract_address_eoa() {
- let (_anvil, _rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, _rpc_url, provider, private_key) = spawn_anvil(None);
let message = "test message";
let message_bytes = alloy::primitives::eip191_hash_message(message.as_bytes());
let signature = sign_message_eip191(message, &private_key);
@@ -377,7 +459,7 @@ mod test {
#[tokio::test]
async fn test_extract_address_eip721() {
- let (_anvil, _rpc_url, provider, _private_key) = spawn_anvil();
+ let (_anvil, _rpc_url, provider, _private_key) = spawn_anvil(None);
sol! {
#[derive(Debug, Serialize)]
struct FooBar {
@@ -425,7 +507,7 @@ mod test {
#[tokio::test]
async fn test_extract_address_erc1271() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let contract_address = deploy_contract(
&rpc_url,
&private_key,
@@ -446,9 +528,7 @@ mod test {
#[tokio::test]
async fn test_extract_address_erc6492() {
- env_logger::init();
-
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let create2_factory_address =
deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
let message = "test message";
@@ -501,7 +581,7 @@ mod test {
}
#[tokio::test]
async fn eoa_pass() {
- let (_anvil, _rpc_url, provider, _private_key) = spawn_anvil();
+ let (_anvil, _rpc_url, provider, _private_key) = spawn_anvil(None);
let private_key = SigningKey::random(&mut rand::thread_rng());
let message = "xxx";
@@ -520,7 +600,7 @@ mod test {
#[tokio::test]
async fn eoa_wrong_signature() {
- let (_anvil, _rpc_url, provider, _private_key) = spawn_anvil();
+ let (_anvil, _rpc_url, provider, _private_key) = spawn_anvil(None);
let private_key = SigningKey::random(&mut rand::thread_rng());
let message = "xxx";
@@ -538,7 +618,7 @@ mod test {
#[tokio::test]
async fn typed_data() {
- let (_anvil, _rpc_url, provider, _private_key) = spawn_anvil();
+ let (_anvil, _rpc_url, provider, _private_key) = spawn_anvil(None);
sol! {
#[derive(Debug, Serialize)]
@@ -603,7 +683,7 @@ mod test {
#[tokio::test]
async fn eoa_wrong_address() {
- let (_anvil, _rpc_url, provider, _private_key) = spawn_anvil();
+ let (_anvil, _rpc_url, provider, _private_key) = spawn_anvil(None);
let private_key = SigningKey::random(&mut rand::thread_rng());
let message = "xxx";
@@ -621,7 +701,7 @@ mod test {
#[tokio::test]
async fn eoa_wrong_message() {
- let (_anvil, _rpc_url, provider, _private_key) = spawn_anvil();
+ let (_anvil, _rpc_url, provider, _private_key) = spawn_anvil(None);
let private_key = SigningKey::random(&mut rand::thread_rng());
let message = "xxx";
@@ -639,7 +719,7 @@ mod test {
#[tokio::test]
async fn erc1271_pass() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let contract_address = deploy_contract(
&rpc_url,
&private_key,
@@ -663,7 +743,7 @@ mod test {
#[tokio::test]
async fn erc1271_wrong_signature() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let contract_address = deploy_contract(
&rpc_url,
&private_key,
@@ -688,7 +768,7 @@ mod test {
#[tokio::test]
async fn erc1271_wrong_signer() {
- let (anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let contract_address = deploy_contract(
&rpc_url,
&private_key,
@@ -714,7 +794,7 @@ mod test {
#[tokio::test]
async fn erc1271_wrong_contract_address() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let mut contract_address = deploy_contract(
&rpc_url,
&private_key,
@@ -740,7 +820,7 @@ mod test {
#[tokio::test]
async fn erc1271_wrong_message() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let contract_address = deploy_contract(
&rpc_url,
&private_key,
@@ -766,10 +846,12 @@ mod test {
env!("OUT_DIR"),
"/../../../../.foundry/forge/out/Erc1271Mock.sol/Erc1271Mock.bytecode"
));
+
const ERC6492_MAGIC_BYTES: [u16; 16] = [
0x6492, 0x6492, 0x6492, 0x6492, 0x6492, 0x6492, 0x6492, 0x6492, 0x6492, 0x6492, 0x6492,
0x6492, 0x6492, 0x6492, 0x6492, 0x6492,
];
+
sol! {
contract Erc1271Mock {
address owner_eoa;
@@ -824,7 +906,7 @@ mod test {
#[tokio::test]
async fn erc6492_pass() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let create2_factory_address =
deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
@@ -846,7 +928,7 @@ mod test {
#[tokio::test]
async fn erc6492_wrong_signature() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let create2_factory_address =
deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
@@ -870,7 +952,7 @@ mod test {
#[tokio::test]
async fn erc6492_wrong_signer() {
- let (anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let create2_factory_address =
deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
@@ -896,7 +978,7 @@ mod test {
#[tokio::test]
async fn erc6492_wrong_contract_address() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let create2_factory_address =
deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
@@ -922,7 +1004,7 @@ mod test {
#[tokio::test]
async fn erc6492_wrong_message() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let create2_factory_address =
deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
@@ -943,9 +1025,10 @@ mod test {
.is_valid(),
);
}
+
#[tokio::test]
async fn test_erc6492_signature_with_invalid_factory() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let create2_factory_address =
deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
let message = "test message";
@@ -970,7 +1053,7 @@ mod test {
#[tokio::test]
async fn test_erc6492_signature_with_empty_factory_calldata() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let create2_factory_address =
deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
let message = "test message";
@@ -993,7 +1076,7 @@ mod test {
#[tokio::test]
async fn test_erc6492_signature_with_mismatched_lengths() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let create2_factory_address =
deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
let message = "test message";
@@ -1018,7 +1101,7 @@ mod test {
#[tokio::test]
async fn test_erc6492_signature_with_invalid_magic_bytes() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let create2_factory_address =
deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
let message = "test message";
@@ -1042,10 +1125,57 @@ mod test {
assert!(matches!(result, Ok(addr) if addr != predeploy_address));
}
+ #[tokio::test]
+ #[cfg(any(test, feature = "reqwest"))]
+ async fn erc6492_deployed_smart_wallet() {
+ // contract deployed at 0x3e5608baE5e195F7EB12792f0Ba1aEf911F364E9 on base sepolia
+ let (_anvil, _rpc_url, provider, _private_key) =
+ spawn_anvil(Some("https://sepolia.base.org"));
+
+ let message = "Example Message";
+ let hash_message: FixedBytes<32> = eip191_hash_message(message.as_bytes());
+ let signature:Bytes = alloy::hex::decode("0x0000000000000000000000000000000000000000000000000000000000000020000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000400000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000000c000000000000000000000000000000000000000000000000000000000000001200000000000000000000000000000000000000000000000000000000000000017000000000000000000000000000000000000000000000000000000000000000111d8c5e6edcd92aeac9ee215372ebf3264300a429ae6dd79f93d71781d1a397d747b83a603182db717314be83bf9ae0d18d0652294a09d8d472a0cdd6f4a11c80000000000000000000000000000000000000000000000000000000000000025f9351d6932d2b1c9aca4fee013c4ff2672f712f28e9c37330bac4fa19292b8351d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000008f7b2274797065223a22776562617574686e2e676574222c226368616c6c656e6765223a2274337659526847564b6258504a476c506e51676d79495950573079442d356b497675536e497a494d485a4d222c226f726967696e223a2268747470733a2f2f6b6579732d626574612e636f696e626173652e636f6d222c2263726f73734f726967696e223a66616c73657d0000000000000000000000000000000000").unwrap().into();
+
+ // Bytes::from(signature.to_vec());
+ let signer = address!("3e5608baE5e195F7EB12792f0Ba1aEf911F364E9");
+
+ let result = verify_signature(signature, signer, hash_message, provider).await;
+
+ println!("{:?}", result);
+ assert!(result.unwrap().is_valid());
+ }
+
+ #[tokio::test]
+ async fn test_erc6492_signature_with_large_factory_calldata() {
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
+ let create2_factory_address =
+ deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
+ let message = "test message";
+ let message_bytes = message_str_to_bytes(message);
+ let signature = sign_message_eip191(message, &private_key);
+
+ let (_predeploy_address, erc6492_signature) = predeploy_signature(
+ Address::from_private_key(&private_key),
+ create2_factory_address,
+ signature,
+ );
+
+ // Increase the size of factory calldata
+ let factory_calldata_offset = bytes_to_usize(&erc6492_signature[32..64]);
+ let large_calldata = vec![0; 1_000_000]; // 1MB of data
+ let mut new_signature = erc6492_signature[..factory_calldata_offset + 32].to_vec();
+ new_signature.extend_from_slice(&U256::from(large_calldata.len()).to_be_bytes::<32>());
+ new_signature.extend_from_slice(&large_calldata);
+ new_signature.extend_from_slice(&erc6492_signature[factory_calldata_offset + 32..]);
+
+ let result = extract_address(new_signature, message_bytes, provider).await;
+ assert!(result.is_err());
+ }
+
#[tokio::test]
#[ignore]
async fn test_erc6492_signature_with_deployed_contract() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let create2_factory_address =
deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
let message = "test message";
@@ -1068,10 +1198,11 @@ mod test {
let result = extract_address(erc6492_signature, message_bytes, provider).await;
assert_eq!(result.unwrap(), address);
}
+
#[tokio::test]
#[ignore]
async fn test_erc6492_signature_with_different_message() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
+ let (_anvil, rpc_url, provider, private_key) = spawn_anvil(None);
let create2_factory_address =
deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
@@ -1163,30 +1294,26 @@ mod test {
println!("Test completed successfully");
}
+ // Manual test. Paste address, signature, message, and project ID to verify
+ // function
#[tokio::test]
- async fn test_erc6492_signature_with_large_factory_calldata() {
- let (_anvil, rpc_url, provider, private_key) = spawn_anvil();
- let create2_factory_address =
- deploy_contract(&rpc_url, &private_key, CREATE2_CONTRACT, None).await;
- let message = "test message";
- let message_bytes = message_str_to_bytes(message);
- let signature = sign_message_eip191(message, &private_key);
-
- let (_predeploy_address, erc6492_signature) = predeploy_signature(
- Address::from_private_key(&private_key),
- create2_factory_address,
- signature,
- );
+ #[ignore]
+ async fn manual() {
+ let (_anvil, _rpc_url, _provider, _private_key) = spawn_anvil(None);
- // Increase the size of factory calldata
- let factory_calldata_offset = bytes_to_usize(&erc6492_signature[32..64]);
- let large_calldata = vec![0; 1_000_000]; // 1MB of data
- let mut new_signature = erc6492_signature[..factory_calldata_offset + 32].to_vec();
- new_signature.extend_from_slice(&U256::from(large_calldata.len()).to_be_bytes::<32>());
- new_signature.extend_from_slice(&large_calldata);
- new_signature.extend_from_slice(&erc6492_signature[factory_calldata_offset + 32..]);
+ let address = address!("AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA");
+ let message = "xxx";
+ let message_hash = eip191_hash_message(message.as_bytes());
+ let signature = bytes!("aaaa");
- let result = extract_address(new_signature, message_bytes, provider).await;
- assert!(result.is_err());
+ let provider = ReqwestProvider::::new_http(
+ "https://rpc.walletconnect.com/v1?chainId=eip155:1&projectId=xxx"
+ .parse()
+ .unwrap(),
+ );
+ assert!(verify_signature(signature, address, message_hash, provider)
+ .await
+ .unwrap()
+ .is_valid());
}
}
diff --git a/src/test_helpers.rs b/src/test_helpers.rs
index 2229c3e..cbe29d0 100644
--- a/src/test_helpers.rs
+++ b/src/test_helpers.rs
@@ -21,13 +21,20 @@ fn format_foundry_dir(path: &str) -> String {
)
}
-pub fn spawn_anvil() -> (AnvilInstance, String, ReqwestProvider, SigningKey) {
- let anvil = Anvil::at(format_foundry_dir("bin/anvil")).spawn();
- let rpc_url = anvil.endpoint();
- let provider = ReqwestProvider::::new_http(anvil.endpoint_url());
- let private_key = anvil.keys().first().unwrap().clone();
+pub fn spawn_anvil(fork_url: Option<&str>) -> (AnvilInstance, String, ReqwestProvider, SigningKey) {
+ let mut anvil = Anvil::at(format_foundry_dir("bin/anvil"));
+
+ if let Some(fork_url) = fork_url {
+ anvil = anvil.fork(fork_url);
+ }
+
+ let anvil_instance = anvil.spawn();
+
+ let rpc_url = anvil_instance.endpoint();
+ let provider = ReqwestProvider::::new_http(anvil_instance.endpoint_url());
+ let private_key = anvil_instance.keys().first().unwrap().clone();
(
- anvil,
+ anvil_instance,
rpc_url,
provider,
SigningKey::from_bytes(&private_key.to_bytes()).unwrap(),