use hex; use num_bigint::BigUint; use serde::{Deserialize, Serialize}; use tycho_core::{ dto::{Chain as TychoCoreChain, ProtocolComponent}, Bytes, }; use crate::encoding::{ errors::EncodingError, serde_primitives::{biguint_string, biguint_string_option}, }; #[derive(Clone, Default, Debug, Deserialize, Serialize)] pub struct Solution { /// Address of the sender. pub sender: Bytes, /// Address of the receiver. pub receiver: Bytes, /// The token being sold (exact in) or bought (exact out). pub given_token: Bytes, /// Amount of the given token. #[serde(with = "biguint_string")] pub given_amount: BigUint, /// The token being bought (exact in) or sold (exact out). pub checked_token: Bytes, /// False if the solution is an exact input solution. Currently only exact input solutions are /// supported. #[serde(default)] pub exact_out: bool, // If set, it will be applied to expected_amount pub slippage: Option, /// Expected amount of the bought token (exact in) or sold token (exact out). #[serde(with = "biguint_string_option")] pub expected_amount: Option, /// Minimum amount to be checked for the solution to be valid. /// If not set, the check will not be performed. #[serde(with = "biguint_string_option")] pub check_amount: Option, /// List of swaps to fulfill the solution. pub swaps: Vec, // If not set, then the Tycho Router will be used pub router_address: Option, // If set, the corresponding native action will be executed. pub native_action: Option, /// If set to true, the solution will be encoded to be sent directly to the Executor and /// skip the router. The user is responsible for managing necessary approvals and token /// transfers. #[serde(default)] pub direct_execution: bool, } #[derive(Clone, PartialEq, Debug, Deserialize, Serialize)] #[serde(rename_all = "snake_case")] pub enum NativeAction { Wrap, Unwrap, } #[derive(Clone, Debug, Deserialize, Serialize)] pub struct Swap { /// Protocol component from tycho indexer pub component: ProtocolComponent, /// Token being input into the pool. pub token_in: Bytes, /// Token being output from the pool. pub token_out: Bytes, /// Percentage of the amount to be swapped in this operation (for example, 0.5 means 50%) #[serde(default)] pub split: f64, } impl Swap { pub fn new>( component: T, token_in: Bytes, token_out: Bytes, split: f64, ) -> Self { Self { component: component.into(), token_in, token_out, split } } } #[derive(Clone, Debug)] pub struct Transaction { // Address of the contract to call with the calldata pub to: Bytes, // ETH value to be sent with the transaction. pub value: BigUint, // Encoded calldata for the transaction. pub data: Vec, } pub struct EncodingContext { pub receiver: Bytes, pub exact_out: bool, pub router_address: Bytes, } #[derive(Clone, PartialEq, Eq, Hash)] pub struct ChainId(pub u64); #[derive(Clone, PartialEq, Eq, Hash)] pub struct Chain { pub id: ChainId, pub name: String, } impl ChainId { pub fn id(&self) -> u64 { self.0 } } impl From for ChainId { fn from(chain: TychoCoreChain) -> Self { match chain { TychoCoreChain::Ethereum => ChainId(1), TychoCoreChain::ZkSync => ChainId(324), TychoCoreChain::Arbitrum => ChainId(42161), TychoCoreChain::Starknet => ChainId(0), } } } impl From for Chain { fn from(chain: TychoCoreChain) -> Self { Chain { id: chain.into(), name: chain.to_string() } } } impl Chain { pub fn native_token(&self) -> Result { match self.id.id() { 1 => Ok(Bytes::from(hex::decode("0000000000000000000000000000000000000000").map_err( |_| EncodingError::FatalError("Failed to decode native token".to_string()), )?)), _ => Err(EncodingError::InvalidInput(format!( "Native token not set for chain {:?}. Double check the chain is supported.", self.name ))), } } pub fn wrapped_token(&self) -> Result { match self.id.id() { 1 => Ok(Bytes::from(hex::decode("C02aaA39b223FE8D0A0e5C4F27eAD9083C756Cc2").map_err( |_| EncodingError::FatalError("Failed to decode wrapped token".to_string()), )?)), _ => Err(EncodingError::InvalidInput(format!( "Wrapped token not set for chain {:?}. Double check the chain is supported.", self.name ))), } } } mod tests { use super::*; struct MockProtocolComponent { id: String, protocol_system: String, } impl From for ProtocolComponent { fn from(component: MockProtocolComponent) -> Self { ProtocolComponent { id: component.id, protocol_system: component.protocol_system, tokens: vec![], protocol_type_name: "".to_string(), chain: Default::default(), contract_ids: vec![], static_attributes: Default::default(), change: Default::default(), creation_tx: Default::default(), created_at: Default::default(), } } } #[test] fn test_swap_new() { let component = MockProtocolComponent { id: "i-am-an-id".to_string(), protocol_system: "uniswap_v2".to_string(), }; let swap = Swap::new(component, Bytes::from("0x12"), Bytes::from("34"), 0.5); assert_eq!(swap.token_in, Bytes::from("0x12")); assert_eq!(swap.token_out, Bytes::from("0x34")); assert_eq!(swap.component.protocol_system, "uniswap_v2"); assert_eq!(swap.component.id, "i-am-an-id"); } }