use std::collections::HashMap;
use alloy::sol_types::SolValue;
use tycho_common::{models::Chain, Bytes};
use crate::encoding::{
errors::EncodingError,
models::{EncodingContext, Swap},
swap_encoder::SwapEncoder,
};
#[derive(Clone)]
pub struct WethSwapEncoder {
executor_address: Bytes,
native_token_address: Bytes,
}
impl SwapEncoder for WethSwapEncoder {
fn new(
executor_address: Bytes,
chain: Chain,
_config: Option<HashMap<String, String>>,
) -> Result<Self, EncodingError> {
let native_token_address = chain.native_token().address;
Ok(Self { executor_address, native_token_address })
}
fn encode_swap(
&self,
swap: &Swap,
_encoding_context: &EncodingContext,
) -> Result<Vec<u8>, EncodingError> {
let is_wrapping = *swap.token_in() == self.native_token_address;
Ok(is_wrapping.abi_encode_packed())
}
fn executor_address(&self) -> &Bytes {
&self.executor_address
}
fn clone_box(&self) -> Box<dyn SwapEncoder> {
Box::new(self.clone())
}
}
#[cfg(test)]
mod tests {
use alloy::hex::encode;
use tycho_common::models::protocol::ProtocolComponent;
use super::*;
use crate::encoding::evm::utils::write_calldata_to_file;
#[test]
fn test_encode_weth_wrapping() {
let pool =
ProtocolComponent { protocol_system: String::from("weth"), ..Default::default() };
let token_in = Bytes::from("0x0000000000000000000000000000000000000000");
let token_out = Bytes::from("0xc02aaa39b223fe8d0a0e5c4f27ead9083c756cc2");
let swap = Swap::new(pool, token_in.clone(), token_out.clone());
let encoding_context = EncodingContext {
router_address: Some(Bytes::default()),
group_token_in: token_in.clone(),
group_token_out: token_out.clone(),
};
let encoder = WethSwapEncoder::new(
Bytes::from("0x543778987b293C7E8Cf0722BB2e935ba6f4068D4"),
Chain::Ethereum,
None,
)
.unwrap();
let encoded_swap = encoder
.encode_swap(&swap, &encoding_context)
.unwrap();
let hex_swap = encode(&encoded_swap);
assert_eq!(hex_swap, String::from("01").to_lowercase());
write_calldata_to_file("test_encode_weth_wrapping", hex_swap.as_str());
}
#[test]
fn test_encode_weth_unwrapping() {
let pool = ProtocolComponent {
id: String::from(""),
protocol_system: String::from(""),
..Default::default()
};
let token_in = Bytes::from("0xc02aaa39b223fe8d0a0e5c4f27ead9083c756cc2");
let token_out = Bytes::from("0x0000000000000000000000000000000000000000");
let swap = Swap::new(pool, token_in.clone(), token_out.clone());
let encoding_context = EncodingContext {
router_address: Some(Bytes::default()),
group_token_in: token_in.clone(),
group_token_out: token_out.clone(),
};
let encoder = WethSwapEncoder::new(
Bytes::from("0x13aa49bAc059d709dd0a18D6bb63290076a702D7"),
Chain::Ethereum,
None,
)
.unwrap();
let encoded_swap = encoder
.encode_swap(&swap, &encoding_context)
.unwrap();
let hex_swap = encode(&encoded_swap);
assert_eq!(hex_swap, String::from("00").to_lowercase());
write_calldata_to_file("test_encode_weth_unwrapping", hex_swap.as_str());
}
}