Skip to content

Instantly share code, notes, and snippets.

@alisinabh
Created January 23, 2024 18:38
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save alisinabh/fe5f199c803b4679ce31d5e0e54aadb8 to your computer and use it in GitHub Desktop.
Save alisinabh/fe5f199c803b4679ce31d5e0e54aadb8 to your computer and use it in GitHub Desktop.
Elixir Ethers and Uniswap

Uniswap ❤️ Elixir

Mix.install([
  {:ethers, "~> 0.2.2"},
  {:uniswap, github: "alisinabh/elixir_uniswap"},
  :ex_secp256k1,
  # For Images
  :kino,
  :ex_url
])

Configuration

Setting up RPC and Signer data

# Testing on Goerli testnet
Application.put_env(:ethereumex, :url, "https://gateway.tenderly.co/public/goerli")
:ok
priv_key = System.get_env("LB_PRIVATE_KEY")
{:ok, [account]} = Ethers.Signer.Local.accounts(private_key: priv_key)

Application.put_env(:ethers, :default_signer, Ethers.Signer.Local)
Application.put_env(:ethers, :default_signer_opts, private_key: priv_key)

account
"0xC68aaEdc8295e8e2307bf4280e233a0Bb62056c9"

Minting Positions

First, pool data is required for creating the position.

usdc = "0xc2a9303EEF2ED1873adE85d2173D7AB6538971D5"
eurc = "0x12C8aAc4454b19CF9FE0918Ea56e6cc9d96bd829"

# 0.3%
pool =
  Uniswap.Contracts.V3Factory.get_pool(usdc, eurc, 3000)
  |> Ethers.call!()
"0xb45c5c5030765faf6c8a0a5c6f9c502689ca8f59"

tick_spacing is needed to calculate lower/upper tick bounds for creating a position.

It the steps which ticks are considered valid. (e.g. 10 means ticks that are divisable to 10 are valid)

tick_spacing =
  Uniswap.Contracts.V3Pool.tick_spacing()
  |> Ethers.call!(to: pool)
60

Calculate ticks using floating point prices (Unsafe Math)

lower_tick = Uniswap.Price.Utils.price_to_tick(0.90, tick_spacing, 6, 6)
upper_tick = Uniswap.Price.Utils.price_to_tick(1.25, tick_spacing, 6, 6)

{lower_tick, upper_tick}
{-1080, 2220}

View balances of our ERC20 tokens

calls = [
  {Ethers.Contracts.ERC20.balance_of(account), to: usdc},
  {Ethers.Contracts.ERC20.balance_of(account), to: eurc}
]

Ethers.Multicall.aggregate3(calls)
|> Ethers.call!()
|> Ethers.Multicall.decode(calls)
[true: 99998819456433, true: 99999777595699]

If not already done, we need to approve the NonfungiblePositionManager contract to spend our tokens using Ethers.Contracts.ERC20.approve/3.

Now we can mint a position using NonfungiblePositionManager.

# Mint tuple: {token_0, token_1, fee, tick_lower, tick_upper, amount_0, amount_1, 
#                min_amount_0, min_amount_1, recepient, deadline}

usdc_amount = trunc(100 * 1.09 * 10 ** 6)
eurc_amount = trunc(100 * 10 ** 6)

Uniswap.Contracts.NonfungiblePositionManager.mint(
  {eurc, usdc, 3000, lower_tick, upper_tick, usdc_amount, eurc_amount, 0, 0, account,
   1_000_000_000_000}
)

# |> Ethers.send!(from: account, max_fee_per_gas: 1297000, max_priority_fee_per_gas: 1297000)
#Ethers.TxData<
  function mint(
    (address,address,uint24,int24,int24,uint256,uint256,uint256,uint256,address,uint256) params {"0x12c8aac4454b19cf9fe0918ea56e6cc9d96bd829",
     "0xc2a9303eef2ed1873ade85d2173d7ab6538971d5", 3000, -1080, 2220, 109000000, 100000000, 0, 0,
     "0xc68aaedc8295e8e2307bf4280e233a0bb62056c9", 1000000000000}
  ) payable returns (
    uint256 tokenId,
    uint128 liquidity,
    uint256 amount0,
    uint256 amount1
  )
  default_address: "0xC36442b4a4522E871399CD717aBDD847Ab11FE88"
>

Get the number of open positions that an account has.

Uniswap.Contracts.NonfungiblePositionManager.balance_of(account)
|> Ethers.call!()
3

Get a position token number (ERC721)

Uniswap.Contracts.NonfungiblePositionManager.token_of_owner_by_index(account, 2)
|> Ethers.call!()
91603

Get position data (Liquidity, Tick bounds, etc.)

Uniswap.Contracts.NonfungiblePositionManager.positions(91603)
|> Ethers.call!()
[0, "0x0000000000000000000000000000000000000000", "0x12c8aac4454b19cf9fe0918ea56e6cc9d96bd829",
 "0xc2a9303eef2ed1873ade85d2173d7ab6538971d5", 3000, -1080, 2220, 994503822,
 5251594197711366159301249239417349, 4095136457040771698219644190665784, 0, 0]

Collecting Fees

Bonus: See how much unclaimed fees a position has.

To collect the fees, just replace call with send

max = Ethers.Types.max({:uint, 128})

Uniswap.Contracts.NonfungiblePositionManager.collect({91603, account, max, max})
|> Ethers.call!(from: account, max_fee_per_gas: 1_297_000, max_priority_fee_per_gas: 1_297_000)
[0, 0]

Increasing/Decreasing Liquidity

# Increase Params: {token_id, amount_0, amount_1, min_amount_0, min_amount_1, deadline}

Uniswap.Contracts.NonfungiblePositionManager.increase_liquidity(
  {91603, 2_000_000, 2_000_000, 10, 10, 10_000_000_000_000}
)
|> Ethers.call!(from: account)
[21063762, 1356406, 2000000]
# Decrease Params: {token_id, liquidity_to_remove, min_amount_0, min_amount_1, deadline}

Uniswap.Contracts.NonfungiblePositionManager.decrease_liquidity(
  {91529, 10000, 10, 10, 10_000_000_000_000}
)
|> Ethers.call!(from: account)
[643, 949]

Getting Current price of the pool

[sqrt_price | _] =
  Uniswap.Contracts.V3Pool.slot0()
  |> Ethers.call!(to: pool)
[82586205569788209757413809326, 830, 0, 1, 1, 0, true]
Uniswap.Price.Utils.from_sqrt_x96(sqrt_price, 6, 6)
1.0865653669376387

NFT Position Image

url_data =
  Uniswap.Contracts.NonfungiblePositionManager.token_uri(91603)
  |> Ethers.call!()
  |> URL.new!()
  |> Map.get(:parsed_path)
  |> Map.get(:data)
  |> Jason.decode!()
  |> IO.inspect()

Kino.HTML.new("<img src=\"#{url_data["image"]}\"/>")
%{
  "description" => "This NFT represents a liquidity position in a Uniswap V3 USDC-EURC pool. The owner of this NFT can modify or redeem the position.\n\nPool Address: 0xb45c5c5030765faf6c8a0a5c6f9c502689ca8f59\nUSDC Address: 0xc2a9303eef2ed1873ade85d2173d7ab6538971d5\nEURC Address: 0x12c8aac4454b19cf9fe0918ea56e6cc9d96bd829\nFee Tier: 0.3%\nToken ID: 91603\n\n⚠️ DISCLAIMER: Due diligence is imperative when assessing this NFT. Make sure token addresses match the expected tokens, as token symbols may be imitated.",
  "image" => "" <> ...,
  "name" => "Uniswap - 0.3% - USDC/EURC - 0.89763<>1.2486"
}

Final Words

  • Unsafe vs Safe math in Elixir
  • Increasing Gas efficiency with a custom contract
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment