Large Withdrawals

Learn how to optimize withdrawals to limit slippage.

At the end of this document you should be able to utilize your own swap instructions, from 0x or Bebop for example, during your Autopool share redemption. For an example script of the below: https://github.com/Tokemak/v2-large-withdraw-example

Autopools allow you to deposit and withdraw in the same asset, the base asset,and to do so on-demand. This means when your withdrawal is large enough to require pulling assets from the underlying LP markets you may incur slippage from swapping back into the Autopool's base asset.

By default, the Autopilot system is configured to be able to swap out of these assets all on-chain with no additional work or information needed. However the quality of trade-execution back into the base asset is limited by the liquidity present in the configured routes that are on-chain. We'll outline here how to provide your own routes to optimize withdrawal execution and to set limit slippage controls.

Determining Swap Amounts

The first step in providing your own routes is to make a rough estimate as to the tokens and amounts that the Autopool will attempt to swap out of. For a TL;DR version, we have provided a package that will give this information for an Autopool and share amount: https://www.npmjs.com/package/@tokemak/autopilot-swap-route-calc If you're able to use the package then you can skip the following section. However for those that wish to calculate these tokens manually, read on.

Calculating Manually

If you wish to calculate these values on your own, you should inspect the logic that is performed during the redeem()operation on the Autopool: https://github.com/Tokemak/v2-core-pub/blob/main/src/vault/AutopoolETH.sol#L357 Some parts to highlight:

  • One cannot receive more than assets than the shares valued using withdrawal pricing

  • If there are enough assets in idle to cover the value of the shares, no swaps are needed

  • The destinations used to pull assets are picked from the withdrawal queue on the Autopool: `getWithdrawalQueue()`

Handling Swaps During Redemption

As with calculating the amounts to swap, we have an interface to get optimal swap executions as well. This is via an API that under the hood utilizes DEX aggregators such as 0x and Bebop. To call:

POST: https://dynamic-swap-routes.tokemaklabs.xyz

{
    "chainId": 1,
    "systemName": "gen3",
    "slippageBps": 10,
    "tokensToLiquidate": [{
        "token": "0x....",
        "amount": "100000000000000000",
        "buyToken": "0xc02aaa39b223fe8d0a0e5c4f27ead9083c756cc2"
    }]
}

Where the tokensToLiquidate is the output from the package @tokemak/autopilot-swap-route-calc package above. systemNamestays "gen3".

The resultsfrom the API output is what will be later passed into our Router for execution:

{ 
    "success": true, 
    "results": [ { 
        "fromToken": "0x7f39c581f595b53c5cb19bd0b3f8da6c935e2ca0", 
        "toToken": "0xc02aaa39b223fe8d0a0e5c4f27ead9083c756cc2", 
        "target": "0x39dd8eff184B860eE7FFe3676eB5102e29a57329", 
        "data": "0x..." 
    } ] 
}

Custom Swap Handling

It's possible to use your own contracts as well during the swapping and redemption process instead our API and swapper contracts. To illustrate what would need to be done lets look at the execution flow.

During the redeem()process, our swapper contracts are engaged 0..n times based on the amount of shares being redeemed and the Destinations/markets we need to pull from to satisfy. For example:

  • User is redeeming 100 Autopool shares worth 102 ETH (idle only has 5 ETH so it can't cover)

  • The Autopool has 50 ETH worth of assets deployed to the first Destination in the withdrawal queue and 200 ETH worth of assets deployed to the second Destination in the withdrawal queue.

  • The first Destination is a wstETH/WETH and the second a pxETH/rETH

  • Lets assume all assets are priced at 1 ETH. We would expect our swapper contracts to be engaged 3 times:

    • Swap 25 wstETH into WETH

      • This swaps to WETH because the asset() on the Autopool is WETH

      • Since the second token in this Destination is already WETH, it is skipped for swapping

    • Swap 26 pxETH into WETH

    • Swap 26 rETH into WETH

  • We exhausted the 50 ETH worth of assets from the first Destination and then moved to the second to get the remainder

Using the output from our API as an example you'd construct an array of the fromToken,toToken,target,data for each of three tokens above, wstETH, pxETH, and rETH:

  • targetis the contract you want us to call

  • datais the call data you'd want us to use to call your target

  • Before your function is called the assets to be swapped will be transferred to the target

  • Before the execution of the function ends, the contract should transfer the new assets back to the msg.sender.

Constructing the Final Transaction

Redemption with custom swap routes is only possible through our Autopilot Router multicall contract (see Contract Addresses). There are multiple ways to construct the call but ultimately you are attempting give the Router access to your tokens (through approvals, sending, or having the Router pull them) and calling redeemWithRoutes()

It is very important not to leave tokens in the Router contract at the end of execution

Last updated

Was this helpful?