use alloy::consensus::{
transaction::RlpEcdsaEncodableTx, SignableTransaction, Signed, Transaction, Typed2718,
};
use alloy::primitives::PrimitiveSignature as Signature;
use alloy::primitives::{keccak256, Address, Bytes, ChainId, TxKind, U256};
use alloy::rlp::{BufMut, Decodable, Encodable, Header};
use alloy::rpc::types::TransactionInput;
use serde::{Deserialize, Serialize};
use crate::network::tx_type::TxType;
pub use self::meta::{Eip712Meta, PaymasterParams};
pub use self::utils::{hash_bytecode, BytecodeHashError};
mod meta;
mod signing;
mod utils;
#[derive(Clone, Debug, Default, PartialEq, Eq, Hash, Serialize, Deserialize)]
#[serde(rename_all = "camelCase")]
#[doc(
alias = "Eip712Transaction",
alias = "TransactionEip712",
alias = "Eip712Tx"
)]
pub struct TxEip712 {
#[serde(with = "alloy::serde::quantity")]
pub chain_id: ChainId,
pub nonce: U256,
#[serde(with = "alloy::serde::quantity")]
pub gas: u64,
#[serde(with = "alloy::serde::quantity")]
pub max_fee_per_gas: u128,
#[serde(with = "alloy::serde::quantity")]
pub max_priority_fee_per_gas: u128, pub to: Address,
pub from: Address,
pub value: U256,
pub input: Bytes,
#[serde(flatten)]
pub eip712_meta: Option<Eip712Meta>,
}
impl TxEip712 {
pub const fn effective_gas_price(&self, base_fee: Option<u64>) -> u128 {
match base_fee {
None => self.max_fee_per_gas,
Some(base_fee) => {
let tip = self.max_fee_per_gas.saturating_sub(base_fee as u128);
if tip > self.max_priority_fee_per_gas {
self.max_priority_fee_per_gas + base_fee as u128
} else {
self.max_fee_per_gas
}
}
}
}
pub(crate) fn encode_with_signature(
&self,
signature: &Signature,
out: &mut dyn BufMut,
) {
out.put_u8(self.tx_type() as u8);
self.encode_with_signature_fields(signature, out);
}
#[doc(hidden)]
pub fn decode_signed_fields(buf: &mut &[u8]) -> alloy::rlp::Result<Signed<Self>> {
let header = Header::decode(buf)?;
if !header.list {
return Err(alloy::rlp::Error::UnexpectedString);
}
let original_len = buf.len();
let nonce = Decodable::decode(buf)?;
let max_priority_fee_per_gas = Decodable::decode(buf)?;
let max_fee_per_gas = Decodable::decode(buf)?;
let gas = Decodable::decode(buf)?;
let to = Decodable::decode(buf)?;
let value = Decodable::decode(buf)?;
let input = Decodable::decode(buf)?;
let signature = Signature::decode_rlp_vrs(buf, bool::decode)?;
let chain_id = Decodable::decode(buf)?;
let from = Decodable::decode(buf)?;
let eip712_meta = Decodable::decode(buf)?;
let tx = Self {
chain_id,
nonce,
gas,
max_fee_per_gas,
max_priority_fee_per_gas,
to,
from,
value,
input,
eip712_meta: Some(eip712_meta),
};
let signed = tx.into_signed(signature);
if buf.len() + header.payload_length != original_len {
return Err(alloy::rlp::Error::ListLengthMismatch {
expected: header.payload_length,
got: original_len - buf.len(),
});
}
Ok(signed)
}
pub(crate) fn fields_len(&self) -> usize {
self.nonce.length()
+ self.max_priority_fee_per_gas.length()
+ self.max_fee_per_gas.length()
+ self.gas.length()
+ self.to.length()
+ self.value.length()
+ self.input.length()
+ self.chain_id.length()
+ self.from.length()
+ self
.eip712_meta
.as_ref()
.map(|m| m.length())
.unwrap_or_default()
}
pub(crate) fn encode_with_signature_fields(&self, signature: &Signature, out: &mut dyn BufMut) {
let payload_length = self.fields_len() + signature.rlp_rs_len() + signature.v().length();
let header = Header {
list: true,
payload_length,
};
header.encode(out);
self.nonce.encode(out);
self.max_priority_fee_per_gas.encode(out);
self.max_fee_per_gas.encode(out);
self.gas.encode(out);
self.to.encode(out);
self.value.encode(out);
self.input.0.encode(out);
signature.write_rlp_vrs(out, signature.v());
self.chain_id.encode(out);
self.from.encode(out);
if let Some(eip712_meta) = &self.eip712_meta {
eip712_meta.encode(out);
}
}
pub(crate) fn encoded_length(&self, signature: &Signature) -> usize {
let payload_length = self.fields_len() + signature.rlp_rs_len() + signature.v().length();
alloy::rlp::length_of_length(payload_length) + payload_length
}
#[doc(alias = "transaction_type")]
pub(crate) const fn tx_type(&self) -> TxType {
TxType::Eip712
}
}
impl Typed2718 for TxEip712 {
fn ty(&self) -> u8 {
self.tx_type() as u8
}
}
impl Transaction for TxEip712 {
fn chain_id(&self) -> Option<ChainId> {
Some(self.chain_id)
}
fn nonce(&self) -> u64 {
(self.nonce % U256::from(u64::MAX)).try_into().unwrap()
}
fn gas_limit(&self) -> u64 {
self.gas
}
fn gas_price(&self) -> Option<u128> {
None
}
fn to(&self) -> Option<Address> {
self.to.into()
}
fn is_create(&self) -> bool {
matches!(self.kind(), TxKind::Create)
}
fn value(&self) -> U256 {
self.value
}
fn input(&self) -> &Bytes {
&self.input
}
fn max_fee_per_gas(&self) -> u128 {
self.max_fee_per_gas
}
fn max_priority_fee_per_gas(&self) -> Option<u128> {
Some(self.max_priority_fee_per_gas)
}
fn max_fee_per_blob_gas(&self) -> Option<u128> {
None
}
fn priority_fee_or_price(&self) -> u128 {
todo!()
}
fn access_list(&self) -> Option<&alloy::rpc::types::AccessList> {
None
}
fn blob_versioned_hashes(&self) -> Option<&[alloy::primitives::B256]> {
None
}
fn authorization_list(&self) -> Option<&[alloy::eips::eip7702::SignedAuthorization]> {
None
}
fn kind(&self) -> TxKind {
self.to.into()
}
fn effective_gas_price(&self, base_fee: Option<u64>) -> u128 {
self.effective_gas_price(base_fee)
}
fn is_dynamic_fee(&self) -> bool {
false
}
}
impl SignableTransaction<Signature> for TxEip712 {
fn set_chain_id(&mut self, chain_id: ChainId) {
self.chain_id = chain_id;
}
fn encode_for_signing(&self, out: &mut dyn alloy::rlp::BufMut) {
out.put_u8(0x19);
out.put_u8(0x01);
out.put_slice(self.domain_hash().as_slice());
out.put_slice(self.eip712_hash_struct().as_slice());
}
fn payload_len_for_signature(&self) -> usize {
2 + 32 + 32
}
fn into_signed(self, signature: Signature) -> Signed<Self> {
let mut buf = [0u8; 64];
buf[..32].copy_from_slice(self.signature_hash().as_slice());
buf[32..].copy_from_slice(keccak256(signature.as_bytes()).as_slice());
let hash = keccak256(buf);
Signed::new_unchecked(self, signature, hash)
}
}
impl RlpEcdsaEncodableTx for TxEip712 {
#[doc = " Calculate the encoded length of the transaction\'s fields, without a RLP"]
#[doc = " header."]
fn rlp_encoded_fields_length(&self) -> usize {
self.rlp_encoded_length()
}
#[doc = " Encodes only the transaction\'s fields into the desired buffer, without"]
#[doc = " a RLP header."]
fn rlp_encode_fields(&self, out: &mut dyn BufMut) {
self.rlp_encode(out);
}
}
impl From<TxEip712> for alloy::rpc::types::transaction::TransactionRequest {
fn from(tx: TxEip712) -> Self {
Self {
transaction_type: Some(tx.tx_type() as u8),
chain_id: Some(tx.chain_id),
nonce: Some((tx.nonce % U256::from(u64::MAX)).try_into().unwrap()), gas: Some(tx.gas),
max_fee_per_gas: Some(tx.max_fee_per_gas),
max_priority_fee_per_gas: Some(tx.max_priority_fee_per_gas),
to: Some(tx.to.into()),
from: Some(tx.from),
value: Some(tx.value),
input: TransactionInput::new(tx.input),
access_list: None,
blob_versioned_hashes: None,
max_fee_per_blob_gas: None,
gas_price: None,
sidecar: None,
authorization_list: None,
}
}
}
#[cfg(test)]
mod tests {
use std::str::FromStr;
use crate::network::unsigned_tx::eip712::{Eip712Meta, PaymasterParams};
use super::TxEip712;
use alloy::consensus::SignableTransaction;
use alloy::hex::FromHex;
use alloy::primitives::{
address, hex, Address, Bytes, FixedBytes, PrimitiveSignature as Signature, B256, U256,
};
#[test]
fn decode_eip712_tx() {
let encoded = hex::decode("f8b701800b0c940754b07d1ea3071c3ec9bd86b2aa6f1a59a514980a8301020380a0635f9ee3a1523de15fc8b72a0eea12f5247c6b6e2369ed158274587af6496599a030f7c66d1ed24fca92527e6974b85b07ec30fdd5c2d41eae46966224add965f982010e9409a6aa96b9a17d7f7ba3e3b19811c082aba9f1e304e1a0020202020202020202020202020202020202020202020202020202020202020283010203d694000000000000000000000000000000000000000080").unwrap();
let signed_tx = TxEip712::decode_signed_fields(&mut &encoded[..]).unwrap();
let tx = signed_tx.tx();
assert_eq!(tx.chain_id, 270);
assert_eq!(tx.nonce, U256::from(1));
assert_eq!(tx.gas, 12);
assert_eq!(tx.max_fee_per_gas, 11);
assert_eq!(tx.max_priority_fee_per_gas, 0);
assert_eq!(tx.to, address!("0754b07d1ea3071c3ec9bd86b2aa6f1a59a51498"));
assert_eq!(
tx.from,
address!("09a6aa96b9a17d7f7ba3e3b19811c082aba9f1e3")
);
assert_eq!(tx.value, U256::from(10));
assert_eq!(tx.input, Bytes::from_hex("0x010203").unwrap());
assert_eq!(
tx.eip712_meta,
Some(Eip712Meta {
gas_per_pubdata: U256::from(4),
factory_deps: vec![Bytes::from_hex(
"0x0202020202020202020202020202020202020202020202020202020202020202"
)
.unwrap()],
custom_signature: Some(Bytes::from_hex("0x010203").unwrap()),
paymaster_params: Some(PaymasterParams {
paymaster: address!("0000000000000000000000000000000000000000"),
paymaster_input: Bytes::from_hex("0x").unwrap()
}),
})
);
}
#[test]
fn decode_eip712_tx_with_paymaster() {
let encoded = hex::decode("f9036580843b9aca00843b9aca0083989680949c1a3d7c98dbf89c7f5d167f2219c29c2fe775a780b903045abef77a000000000000000000000000000000000000000000000000000000000000004000000000000000000000000000000000000000000000000000000000000002e0000000000000000000000000000000000000000000000000000000000000008000000000000000000000000051ef809ffd89cf8056d4c17f0aff1b6f8257eb6000000000000000000000000000000000000000000000000000000000000001f4000000000000000000000000000000000000000000000000000000000000000500000000000000000000000000000000000000000000000000000000000001e10100000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000000000040000000000000000000000000000000000000000000000000000000000000010f0000000000000000000000000000000000000000000000000000000000000001000000000000000000000000000000000000000000000000000000000000002000000000000000000000000078cad996530109838eb016619f5931a03250489a000000000000000000000000aaf5f437fb0524492886fba64d703df15bf619ae000000000000000000000000000000000000000000000000000000000000010f00000000000000000000000000000000000000000000000000000000000000a000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000064a41368620000000000000000000000000000000000000000000000000000000000000020000000000000000000000000000000000000000000000000000000000000000568656c6c6f0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000080808082010f94b2a6e81272904caf680008078feb36336d9376b482c350c080d89499e12239cbf8112fbb3f7fd473d0558031abcbb5821234").unwrap();
let signed_tx = TxEip712::decode_signed_fields(&mut &encoded[..]).unwrap();
let tx = signed_tx.tx();
assert_eq!(tx.chain_id, 271);
assert_eq!(tx.nonce, U256::from(0));
assert_eq!(tx.gas, 10000000);
assert_eq!(tx.max_fee_per_gas, 1000000000);
assert_eq!(tx.max_priority_fee_per_gas, 1000000000);
assert_eq!(tx.to, address!("9c1a3d7c98dbf89c7f5d167f2219c29c2fe775a7"));
assert_eq!(
tx.from,
address!("b2a6e81272904caf680008078feb36336d9376b4")
);
assert_eq!(tx.value, U256::from(0));
assert_eq!(tx.input, Bytes::from_hex("5abef77a000000000000000000000000000000000000000000000000000000000000004000000000000000000000000000000000000000000000000000000000000002e0000000000000000000000000000000000000000000000000000000000000008000000000000000000000000051ef809ffd89cf8056d4c17f0aff1b6f8257eb6000000000000000000000000000000000000000000000000000000000000001f4000000000000000000000000000000000000000000000000000000000000000500000000000000000000000000000000000000000000000000000000000001e10100000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000000000040000000000000000000000000000000000000000000000000000000000000010f0000000000000000000000000000000000000000000000000000000000000001000000000000000000000000000000000000000000000000000000000000002000000000000000000000000078cad996530109838eb016619f5931a03250489a000000000000000000000000aaf5f437fb0524492886fba64d703df15bf619ae000000000000000000000000000000000000000000000000000000000000010f00000000000000000000000000000000000000000000000000000000000000a000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000064a41368620000000000000000000000000000000000000000000000000000000000000020000000000000000000000000000000000000000000000000000000000000000568656c6c6f00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000").unwrap());
assert_eq!(
tx.eip712_meta,
Some(Eip712Meta {
gas_per_pubdata: U256::from(50000),
factory_deps: vec![],
custom_signature: Some(Bytes::from_hex("0x").unwrap()),
paymaster_params: Some(PaymasterParams {
paymaster: address!("99E12239CBf8112fBB3f7Fd473d0558031abcbb5"),
paymaster_input: Bytes::from_hex("0x1234").unwrap()
}),
})
);
}
#[test]
fn test_eip712_tx() {
let eip712_meta = Eip712Meta {
gas_per_pubdata: U256::from(4),
factory_deps: vec![vec![2; 32].into()],
custom_signature: Some(vec![].into()),
paymaster_params: None,
};
let tx = TxEip712 {
chain_id: 270,
from: Address::from_str("0xe30f4fb40666753a7596d315f2f1f1d140d1508b").unwrap(),
to: Address::from_str("0x82112600a140ceaa9d7da373bb65453f7d99af4b").unwrap(),
nonce: U256::from(1),
value: U256::from(10),
gas: 12,
max_fee_per_gas: 11,
max_priority_fee_per_gas: 0,
input: vec![0x01, 0x02, 0x03].into(),
eip712_meta: Some(eip712_meta),
};
let expected_signature_hash = FixedBytes::<32>::from_str(
"0xfc76820a67d9b1b351f2ac661e6d2bcca1c67508ae4930e036f540fa135875fe",
)
.unwrap();
assert_eq!(tx.signature_hash(), expected_signature_hash);
let signature = Signature::from_str("0x3faf83b5451ad3001f96f577b0bb5dfcaa7769ab11908f281dc6b15c45a3986f0325197832aac9a7ab2f5a83873834d457e0d22c1e72377d45364c6968f8ac3b1c").unwrap();
let recovered_signer = signature
.recover_address_from_prehash(&tx.signature_hash())
.unwrap();
assert_eq!(recovered_signer, tx.from);
let mut buf = Vec::new();
tx.encode_with_signature_fields(&signature, &mut buf);
let decoded = TxEip712::decode_signed_fields(&mut &buf[..]).unwrap();
assert_eq!(decoded, tx.into_signed(signature));
let expected_hash =
B256::from_str("0xb85668399db249d62d06bbc59eace82e01364602fb7159e161ca810ff6ddbbf4")
.unwrap();
assert_eq!(*decoded.hash(), expected_hash);
}
#[test]
fn test_eip712_tx_encode_decode_with_paymaster() {
let eip712_meta = Eip712Meta {
gas_per_pubdata: U256::from(4),
factory_deps: vec![vec![2; 32].into()],
custom_signature: Some(vec![].into()),
paymaster_params: Some(PaymasterParams {
paymaster: address!("99E12239CBf8112fBB3f7Fd473d0558031abcbb5"),
paymaster_input: Bytes::from_hex("0x112233").unwrap(),
}),
};
let tx = TxEip712 {
chain_id: 270,
from: Address::from_str("0xe30f4fb40666753a7596d315f2f1f1d140d1508b").unwrap(),
to: Address::from_str("0x82112600a140ceaa9d7da373bb65453f7d99af4b").unwrap(),
nonce: U256::from(1),
value: U256::from(10),
gas: 12,
max_fee_per_gas: 11,
max_priority_fee_per_gas: 0,
input: vec![0x01, 0x02, 0x03].into(),
eip712_meta: Some(eip712_meta),
};
let signature = Signature::from_str("0x3faf83b5451ad3001f96f577b0bb5dfcaa7769ab11908f281dc6b15c45a3986f0325197832aac9a7ab2f5a83873834d457e0d22c1e72377d45364c6968f8ac3b1c").unwrap();
let mut buf = Vec::new();
tx.encode_with_signature_fields(&signature, &mut buf);
let decoded = TxEip712::decode_signed_fields(&mut &buf[..]).unwrap();
assert_eq!(decoded, tx.into_signed(signature));
}
}