tycho-execution 0.301.1

Provides tools for encoding and executing swaps against Tycho router and protocol executors.
Documentation
use std::str::FromStr;

use num_bigint::BigUint;
use tycho_common::{
    models::{protocol::ProtocolComponent, Chain},
    Bytes,
};
use tycho_execution::encoding::{
    evm::{
        encoder_builders::TychoRouterEncoderBuilder,
        swap_encoder::swap_encoder_registry::SwapEncoderRegistry,
    },
    models::{default_token, Solution, Swap},
};

fn main() {
    let user_address = Bytes::from_str("0xcd09f75E2BF2A4d11F3AB23f1389FcC1621c0cc2")
        .expect("Failed to create user address");
    // Initialize the encoder
    let swap_encoder_registry = SwapEncoderRegistry::new(Chain::Ethereum)
        .add_default_encoders(None)
        .expect("Failed to get default SwapEncoderRegistry");
    let encoder = TychoRouterEncoderBuilder::new()
        .chain(Chain::Ethereum)
        .swap_encoder_registry(swap_encoder_registry)
        .build()
        .expect("Failed to build encoder");

    // ------------------- Encode a simple swap -------------------

    // Prepare data to encode. We will encode a simple swap from WETH to USDC
    // First we need to create a swap object
    let weth = Bytes::from_str("0xc02aaa39b223fe8d0a0e5c4f27ead9083c756cc2")
        .expect("Failed to create WETH address");
    let usdc = Bytes::from_str("0xa0b86991c6218b36c1d19d4a2e9eb0ce3606eb48")
        .expect("Failed to create USDC address");

    // The protocol component data comes from tycho-indexer
    let simple_swap = Swap::new(
        ProtocolComponent {
            id: "0xB4e16d0168e52d35CaCD2c6185b44281Ec28C9Dc".to_string(),
            protocol_system: "uniswap_v2".to_string(),
            ..Default::default()
        },
        default_token(weth.clone()),
        default_token(usdc.clone()),
        BigUint::ZERO,
    );
    // Split defines the fraction of the amount to be swapped. A value of 0 indicates 100% of
    // the amount or the total remaining balance. (0 is default, so no need to set it)

    // Then we create a solution object with the previous swap
    let solution = Solution::new(
        user_address.clone(),
        user_address.clone(),
        weth.clone(),
        usdc.clone(),
        BigUint::from_str("1_000000000000000000").expect("Failed to create amount"),
        BigUint::from(1u64),
        vec![simple_swap],
    );

    // Encode the solution
    let encoded_solution = encoder
        .encode_solutions(vec![solution.clone()])
        .expect("Failed to encode router calldata")[0]
        .clone();
    println!(" ====== Simple swap WETH -> USDC ======");
    println!(
        "The simple swap encoded solution should be sent to address {:?} with function signature {:?} and the \
    following encoded swaps: {:?}",
        encoded_solution.interacting_with(),
        encoded_solution.function_signature(),
        hex::encode(encoded_solution.swaps())
    );

    // ------------------- Encode a swap with multiple splits -------------------
    // To illustrate a more complex solution, we will encode a swap from WETH to USDC with multiple
    // splits. Performs a split swap from WETH to USDC though WBTC and DAI using USV2 pools
    //
    //         ┌──(USV2)──> WBTC ───(USV2)──> USDC
    //   WETH ─┤
    //         └──(USV2)──> DAI  ───(USV2)──> USDC
    //

    let wbtc = Bytes::from_str("0x2260fac5e5542a773aa44fbcfedf7c193bc2c599")
        .expect("Failed to create WBTC address");
    let dai = Bytes::from_str("0x6b175474e89094c44da98b954eedeac495271d0f")
        .expect("Failed to create DAI address");

    let swap_weth_dai = Swap::new(
        ProtocolComponent {
            id: "0xA478c2975Ab1Ea89e8196811F51A7B7Ade33eB11".to_string(),
            protocol_system: "uniswap_v2".to_string(),
            ..Default::default()
        },
        default_token(weth.clone()),
        default_token(dai.clone()),
        BigUint::ZERO,
    )
    .with_split(0.5);

    // Split 0 represents the remaining 50%, but to avoid any rounding errors we set this to
    // 0 to signify "the remainder of the WETH value". It should still be very close to 50%
    let swap_weth_wbtc = Swap::new(
        ProtocolComponent {
            id: "0xBb2b8038a1640196FbE3e38816F3e67Cba72D940".to_string(),
            protocol_system: "uniswap_v2".to_string(),
            ..Default::default()
        },
        default_token(weth.clone()),
        default_token(wbtc.clone()),
        BigUint::ZERO,
    );

    let swap_dai_usdc = Swap::new(
        ProtocolComponent {
            id: "0xAE461cA67B15dc8dc81CE7615e0320dA1A9aB8D5".to_string(),
            protocol_system: "uniswap_v2".to_string(),
            ..Default::default()
        },
        default_token(dai.clone()),
        default_token(usdc.clone()),
        BigUint::ZERO,
    );
    let swap_wbtc_usdc = Swap::new(
        ProtocolComponent {
            id: "0x004375Dff511095CC5A197A54140a24eFEF3A416".to_string(),
            protocol_system: "uniswap_v2".to_string(),
            ..Default::default()
        },
        default_token(wbtc.clone()),
        default_token(usdc.clone()),
        BigUint::ZERO,
    );
    let complex_solution = solution.clone().with_swaps(vec![
        swap_weth_dai,
        swap_weth_wbtc,
        swap_dai_usdc,
        swap_wbtc_usdc,
    ]);

    // Encode the solution
    let complex_encoded_solution = encoder
        .encode_solutions(vec![complex_solution])
        .expect("Failed to encode router calldata")[0]
        .clone();

    println!(" ====== Complex split swap WETH -> USDC ======");
    println!(
        "The complex swaps encoded solution should be sent to address {:?} with function signature {:?} and the \
    following encoded swaps: {:?}",
        complex_encoded_solution.interacting_with(),
        complex_encoded_solution.function_signature(),
        hex::encode(complex_encoded_solution.swaps())
    );
}