@version 1 condition triggered_by: transaction, on: add_liquidity(token1_min_amount, token2_min_amount), as: [ token_transfers: ( valid_amounts? = false valid_liquidity? = false user_amounts = get_user_transfers_amount(transaction) if user_transfers.token1 > 0 && user_transfers.token2 > 0 do lp_token_supply = State.get("lp_token_supply", 0) reserves = State.get("reserves", [token1: 0, token2: 0]) final_amounts = nil if lp_token_supply != 0 do # Returns final_amounts.token1 == 0 in case of insufficient funds final_amounts = get_final_amounts(user_amounts, reserves, token1_min_amount, token2_min_amount) else final_amounts = [token1: user_amounts.token1, token2: user_amounts.token2] end if final_amounts.token1 != 0 do valid_amounts? = true pool_balances = get_pool_balances() # Amount = final amounts + potential current balance over reserve token1_amount = final_amounts.token1 + (pool_balances.token1 - reserves.token1) token2_amount = final_amounts.token2 + (pool_balances.token2 - reserves.token2) lp_token_to_mint = get_lp_token_to_mint(token1_amount, token2_amount) valid_liquidity? = lp_token_to_mint > 0 end end valid_amounts? && valid_liquidity? ) ] actions triggered_by: transaction, on: add_liquidity(token1_min_amount, token2_min_amount) do pool_balances = get_pool_balances() user_amounts = get_user_transfers_amount(transaction) lp_token_supply = State.get("lp_token_supply", 0) reserves = State.get("reserves", [token1: 0, token2: 0]) final_amounts = get_final_amounts(user_amounts, reserves, token1_min_amount, token2_min_amount) token1_to_refund = user_amounts.token1 - final_amounts.token1 token2_to_refund = user_amounts.token2 - final_amounts.token2 token1_amount = pool_balances.token1 - reserves.token1 - token1_to_refund token2_amount = pool_balances.token2 - reserves.token2 - token2_to_refund lp_token_to_mint = get_lp_token_to_mint(token1_amount, token2_amount) lp_token_to_mint_bigint = Math.trunc(lp_token_to_mint * 100_000_000) # Remove minimum liquidity if this is the first liquidity if the pool # First liquidity minted and burned on pool creation if lp_token_supply == 0 do lp_token_to_mint_bigint = lp_token_to_mint_bigint - 10 end token_specification = [ aeip: [8, 18, 19], supply: lp_token_to_mint_bigint, token_reference: 0x000064DF14F0C50C0EE7234BD394081FAAFCC53E4A38C712F489E0A53A147EA78716, recipients: [ [to: transaction.address, amount: lp_token_to_mint_bigint] ] ] new_token1_reserve = pool_balances.token1 - token1_to_refund new_token2_reserve = pool_balances.token2 - token2_to_refund State.set("lp_token_supply", lp_token_supply + lp_token_to_mint) State.set("reserves", [token1: new_token1_reserve, token2: new_token2_reserve]) if token1_to_refund > 0 do Contract.add_token_transfer(to: transaction.address, amount: token1_to_refund, token_address: "0000AA56D7FA5AD3524A4EC074A0A3746E38CBC97054BFCB1A9221B348FE3638A0E8") end if token2_to_refund > 0 do if "UCO" == "UCO" do Contract.add_uco_transfer(to: transaction.address, amount: token2_to_refund) else Contract.add_token_transfer(to: transaction.address, amount: token2_to_refund, token_address: "UCO") end end Contract.set_type("token") Contract.set_content(Json.to_string(token_specification)) end condition triggered_by: transaction, on: remove_liquidity(), as: [ token_transfers: ( valid? = false user_amount = get_user_lp_amount(transaction.token_transfers) lp_token_supply = State.get("lp_token_supply", 0) if user_amount > 0 && lp_token_supply > 0 do pool_balances = get_pool_balances() token1_to_remove = (user_amount * pool_balances.token1) / lp_token_supply token2_to_remove = (user_amount * pool_balances.token2) / lp_token_supply valid? = token1_to_remove > 0 && token2_to_remove > 0 end valid? ) ] actions triggered_by: transaction, on: remove_liquidity() do user_amount = get_user_lp_amount(transaction.token_transfers) pool_balances = get_pool_balances() lp_token_supply = State.get("lp_token_supply") token1_to_remove = (user_amount * pool_balances.token1) / lp_token_supply token2_to_remove = (user_amount * pool_balances.token2) / lp_token_supply new_token1_reserve = pool_balances.token1 - token1_to_remove new_token2_reserve = pool_balances.token2 - token2_to_remove State.set("lp_token_supply", lp_token_supply - user_amount) State.set("reserves", [token1: new_token1_reserve, token2: new_token2_reserve]) Contract.set_type("transfer") Contract.add_token_transfer(to: transaction.address, amount: token1_to_remove, token_address: "0000AA56D7FA5AD3524A4EC074A0A3746E38CBC97054BFCB1A9221B348FE3638A0E8") if "UCO" == "UCO" do Contract.add_uco_transfer(to: transaction.address, amount: token2_to_remove) else Contract.add_token_transfer(to: transaction.address, amount: token2_to_remove, token_address: "UCO") end end condition triggered_by: transaction, on: swap(min_to_receive), as: [ token_transfers: ( valid? = false transfer = get_user_transfer(transaction) if transfer != nil do swap = get_swap_infos(transfer.token_address, transfer.amount) valid? = swap.output_amount > 0 && swap.output_amount >= min_to_receive end valid? ) ] actions triggered_by: transaction, on: swap(_min_to_receive) do transfer = get_user_transfer(transaction) swap = get_swap_infos(transfer.token_address, transfer.amount) pool_balances = get_pool_balances() token_to_send = nil if transfer.token_address == "0000AA56D7FA5AD3524A4EC074A0A3746E38CBC97054BFCB1A9221B348FE3638A0E8" do pool_balances = Map.set(pool_balances, "token2", pool_balances.token2 - swap.output_amount) token_to_send = "UCO" else pool_balances = Map.set(pool_balances, "token1", pool_balances.token1 - swap.output_amount) token_to_send = "0000AA56D7FA5AD3524A4EC074A0A3746E38CBC97054BFCB1A9221B348FE3638A0E8" end State.set("reserves", [token1: pool_balances.token1, token2: pool_balances.token2]) Contract.set_type("transfer") if token_to_send == "UCO" do Contract.add_uco_transfer(to: transaction.address, amount: swap.output_amount) else Contract.add_token_transfer(to: transaction.address, amount: swap.output_amount, token_address: token_to_send) end end condition triggered_by: transaction, on: update_code(), as: [ previous_public_key: ( # Pool code can only be updated from the router contract of the dex # Transaction is not yet validated so we need to use previous address # to get the genesis address previous_address = Chain.get_previous_address() Chain.get_genesis_address(previous_address) == 0x00000dd757ac0a67fd619c1d1c400037a38cdb0471e6496807d8b850dc422e5ca3aa ) ] actions triggered_by: transaction, on: update_code() do params = [ "0000AA56D7FA5AD3524A4EC074A0A3746E38CBC97054BFCB1A9221B348FE3638A0E8", "UCO", 0x0000213EBE0109C43A0C04E269771A5C825B9E0740FC6151B6A754813C2E8769AA1A, 0x000064DF14F0C50C0EE7234BD394081FAAFCC53E4A38C712F489E0A53A147EA78716 ] new_code = Contract.call_function(0x00000dd757ac0a67fd619c1d1c400037a38cdb0471e6496807d8b850dc422e5ca3aa, "get_pool_code", params) if Code.is_valid?(new_code) && !Code.is_same?(new_code, contract.code) do Contract.set_type("contract") Contract.set_code(new_code) end end export fun get_ratio(token_address) do reserves = State.get("reserves", [token1: 0, token2: 0]) ratio = 0 token_address = String.to_uppercase(token_address) if reserves.token1 > 0 && reserves.token2 > 0 do if token_address == "0000AA56D7FA5AD3524A4EC074A0A3746E38CBC97054BFCB1A9221B348FE3638A0E8" do ratio = reserves.token2 / reserves.token1 else ratio = reserves.token1 / reserves.token2 end end ratio end export fun get_equivalent_amount(token_address, amount) do reserves = State.get("reserves", [token1: 0, token2: 0]) ratio = 0 token_address = String.to_uppercase(token_address) if reserves.token1 > 0 && reserves.token2 > 0 do if token_address == "0000AA56D7FA5AD3524A4EC074A0A3746E38CBC97054BFCB1A9221B348FE3638A0E8" do ratio = reserves.token2 / reserves.token1 else ratio = reserves.token1 / reserves.token2 end end amount * ratio end export fun get_lp_token_to_mint(token1_amount, token2_amount) do lp_token_supply = State.get("lp_token_supply", 0) reserves = State.get("reserves", [token1: 0, token2: 0]) if lp_token_supply == 0 || reserves.token1 == 0 || reserves.token2 == 0 do # First liquidity Math.sqrt(token1_amount * token2_amount) else mint_amount1 = (token1_amount * lp_token_supply) / reserves.token1 mint_amount2 = (token2_amount * lp_token_supply) / reserves.token2 if mint_amount1 < mint_amount2 do mint_amount1 else mint_amount2 end end end export fun get_swap_infos(token_address, amount) do output_amount = 0 fee = 0 price_impact = 0 reserves = State.get("reserves", [token1: 0, token2: 0]) token_address = String.to_uppercase(token_address) if reserves.token1 > 0 && reserves.token2 > 0 do fee = amount * 0.0025 amount_with_fee = amount - fee market_price = 0 if token_address == "0000AA56D7FA5AD3524A4EC074A0A3746E38CBC97054BFCB1A9221B348FE3638A0E8" do market_price = amount_with_fee * (reserves.token2 / reserves.token1) amount = (amount_with_fee * reserves.token2) / (amount_with_fee + reserves.token1) if amount < reserves.token2 do output_amount = amount end else market_price = amount_with_fee * (reserves.token1 / reserves.token2) amount = (amount_with_fee * reserves.token1) / (amount_with_fee + reserves.token2) if amount < reserves.token1 do output_amount = amount end end if output_amount > 0 do # This check is necessary as there might be some approximation in small decimal calculation if market_price > output_amount do price_impact = ((market_price / output_amount) - 1) * 100 else price_impact = 0 end end end [ output_amount: output_amount, fee: fee, price_impact: price_impact ] end export fun get_remove_amounts(lp_token_amount) do reserves = State.get("reserves", [token1: 0, token2: 0]) lp_token_supply = State.get("lp_token_supply", 0) token1_to_remove = 0 token2_to_remove = 0 if lp_token_supply > 0 && lp_token_amount < lp_token_supply do token1_to_remove = (lp_token_amount * reserves.token1) / lp_token_supply token2_to_remove = (lp_token_amount * reserves.token2) / lp_token_supply end [token1: token1_to_remove, token2: token2_to_remove] end export fun get_pool_infos() do reserves = State.get("reserves", [token1: 0, token2: 0]) [ token1: [ address: "0000AA56D7FA5AD3524A4EC074A0A3746E38CBC97054BFCB1A9221B348FE3638A0E8", reserve: reserves.token1 ], token2: [ address: "UCO", reserve: reserves.token2 ], lp_token: [ address: 0x000064DF14F0C50C0EE7234BD394081FAAFCC53E4A38C712F489E0A53A147EA78716, supply: State.get("lp_token_supply", 0) ], fee: 0.25 ] end fun get_final_amounts(user_amounts, reserves, token1_min_amount, token2_min_amount) do final_token1_amount = 0 final_token2_amount = 0 if reserves.token1 > 0 && reserves.token2 > 0 do token2_ratio = reserves.token2 / reserves.token1 token2_equivalent_amount = user_amounts.token1 * token2_ratio if token2_equivalent_amount <= user_amounts.token2 && token2_equivalent_amount >= token2_min_amount do final_token1_amount = user_amounts.token1 final_token2_amount = token2_equivalent_amount else token1_ratio = reserves.token1 / reserves.token2 token1_equivalent_amount = user_amounts.token2 * token1_ratio if token1_equivalent_amount <= user_amounts.token1 && token1_equivalent_amount >= token1_min_amount do final_token1_amount = token1_equivalent_amount final_token2_amount = user_amounts.token2 end end else # No reserve final_token1_amount = user_amounts.token1 final_token2_amount = user_amounts.token2 end [token1: final_token1_amount, token2: final_token2_amount] end fun get_user_transfers_amount(tx) do contract_address = 0x0000213EBE0109C43A0C04E269771A5C825B9E0740FC6151B6A754813C2E8769AA1A token1_amount = 0 token2_amount = 0 transfers = Map.get(tx.token_transfers, contract_address) uco_amount = Map.get(tx.uco_transfers, contract_address) if uco_amount != nil do transfers = List.prepend(transfers, [token_address: "UCO", amount: uco_amount]) end if List.size(transfers) == 2 do for transfer in transfers do if transfer.token_address == "0000AA56D7FA5AD3524A4EC074A0A3746E38CBC97054BFCB1A9221B348FE3638A0E8" do token1_amount = transfer.amount end if transfer.token_address == "UCO" do token2_amount = transfer.amount end end end [token1: token1_amount, token2: token2_amount] end fun get_user_transfer(tx) do contract_address = 0x0000213EBE0109C43A0C04E269771A5C825B9E0740FC6151B6A754813C2E8769AA1A token_transfer = nil transfers = Map.get(tx.token_transfers, contract_address, []) uco_amount = Map.get(tx.uco_transfers, contract_address) if uco_amount != nil do transfers = List.prepend(transfers, [token_address: "UCO", amount: uco_amount]) end transfer = List.at(transfers, 0) tokens = [ "0000AA56D7FA5AD3524A4EC074A0A3746E38CBC97054BFCB1A9221B348FE3638A0E8", "UCO" ] if List.size(transfers) == 1 && List.in?(tokens, transfer.token_address) do token_transfer = transfer end token_transfer end fun get_user_lp_amount(token_transfers) do lp_token = 0x000064DF14F0C50C0EE7234BD394081FAAFCC53E4A38C712F489E0A53A147EA78716 lp_amount = 0 transfers = Map.get(token_transfers, Chain.get_burn_address(), []) for transfer in transfers do if transfer.token_address == lp_token do lp_amount = transfer.amount end end lp_amount end fun get_pool_balances() do balances = Chain.get_balance(contract.address) token2_balance = 0 if "UCO" == "UCO" do token2_balance = balances.uco else token2_id = [token_address: "UCO", token_id: 0] token2_balance = Map.get(balances.tokens, token2_id, 0) end token1_id = [token_address: "0000AA56D7FA5AD3524A4EC074A0A3746E38CBC97054BFCB1A9221B348FE3638A0E8", token_id: 0] [ token1: Map.get(balances.tokens, token1_id, 0), token2: token2_balance ] end
Content (0 B)
-
Secret shared with 1 key
Encoded secret
911ED81FB23302C50C221F2C86479130B96CA5656CEEF4373744D33EDE422304EB30E5167CB6324FA0C8924386196AC12A8DE420636B05D73799A992
Authorized keys
- 00017877BCF4122095926A49489009649603AB129822A19EF9D573B8FD714911ED7F
Contract recipients (0)
Inputs (0)
Contract inputs (0)
Unspent outputs (2)
-
From 0000818F...061AAt 2023-12-22 16:18:18 UTCAmount 0.00000001 UCO
-
From 0000DB52...53E3At 2023-12-10 08:52:02 UTCAmount 0.00000102 0000AA56...A0E8
Previous public key
00013C877A8F92A684D0A70588AB2FB21C3784F39F67BD15772903FEC6C2A5C0718E
Previous signature
11F557F9D4D69600822B968EBDCA4FE08CEB9303D328EE61297AB7E43DE4E162CBC5E4D39265E301D6D4975B3556EE3B6C3510265C47D664CFEA17EB9EE9280E
Origin signature
30440220094B9D5D074D86A29FFF7271A109896E645525F1E067F754DF32DCDD7F076B0802202CA839EF042937E739173BC641FCAB1D0D2DEF1744C28575A9970223CCD684DA
Proof of work
0101046C39D56B717142B6EE14B0F8B2561ECA458F3D2D12C9977D613F78829419063211F9C21F7BB0D56B6523040A8156BE22B6E9D6332B88ED882574D12AE32F7A45
Proof of integrity
00DC821E90AF6FF820BF490BA31F6E0882F6E8EA33BC305DB11B768E9B69919B1D
Coordinator signature
1101BC53D96B73B59C5171A7DA5861BEAABEC425FD5A1B256FDAF58A24DC272B4A5709B8CDB6AD5717F67645BBDE29A405426465F194C8C2935CB2F7B4B00505
Validator #1 public key
0001B0A94804BF8ECC9897075C6207FF63EF4D339F57A0349888E6B77CD47DB53EF3
Validator #1 signature
108721A192A7ED1FF9DF566EFEDD385B289BF1021F383A3E4736E1BC0F4621429CB2E3FB69196D693AD5E15C905B6DEDE95F1510FE5FD02F67D5C2F3D492E203
Validator #2 public key
000151C124A6211CD402FD1CFE560C5DB51ED0CBEF44B09B21A41206028E7E5942BF
Validator #2 signature
7D038ADF817BE2D1F2D2CC69A8DA376E7F83DED2563CFC84C2840298BB63D1C45C9B6E969C665EC411B00A5E58F95D85AA7721152F527334826A9C22325E1D02