Contract 0x30df229cefa463e991e29d42db0bae2e122b2ac7 2

 
Txn Hash Method
Block
From
To
Value [Txn Fee]
0xb9bda68139833c8a0c3d446d07e65fa2bc3a22528451959d37dd53f8b1c9a09dApprove560014432023-01-28 1:05:083 hrs 28 mins ago0x845a1595deca7e3319a05cd23f9d84fe840640cb IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00003431
0x0e26f0d2e3fa78838d3220080f52619a93ac580757a8715dc33045d9030c22d4Approve559831472023-01-27 23:10:545 hrs 22 mins ago0xf674772acc49eb57960784857b4ef27944a2d94b IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00003925
0xbd1e3779bce6b0ee2dd0162f43a2cb7a049ee228b4bcd0c5e897ab83f0ad445dApprove557920652023-01-27 6:47:3321 hrs 45 mins ago0x69bc6453fc4689b89cb10510b16f70b26170a89e IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00003212
0xdc9257f20594c2bd2fccaa19972086e34e747de39580b0fe3f89272b24fd1c7cApprove557728582023-01-27 4:55:5723 hrs 37 mins ago0xdb453fce1aab3a9c1a82e91717899b36509b4eee IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00003267
0x689a59c11873d315929091c7cbc56fd76dcbd6cceb3238c61d335002945c2791Exchange_underly...557652872023-01-27 4:07:591 day 25 mins ago0x77cde47e1da1e3274ce93f82d210473f48f99d96 IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00007393
0xbfaff2148ad6566b92c2d696ab9e8840b4192cfd75b6a44007a6f3c1738d6277Approve557057812023-01-26 21:52:121 day 6 hrs ago0xdd04bc56e8fa7376b1f848ea3f357645f5fc50ca IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00005606
0x9b6ff25bdcd165a7fbab074a82208fc06b937dba196b2996a9db7fad01183c92Approve557056632023-01-26 21:51:351 day 6 hrs ago0xdd04bc56e8fa7376b1f848ea3f357645f5fc50ca IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00005606
0x01a338e60b77a6bb389f4f2c41a44b0449734f037836eb2d92d6bfd9766a60fbApprove557014912023-01-26 21:26:391 day 7 hrs ago0x407f9321b07b424193023177ac76eee1d9556d54 IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00136308
0x0778309f0f3704d7333b3b1bbd66e8d352606728b54fca5247f19d9c9b130611Approve556628402023-01-26 18:08:051 day 10 hrs ago0x6d70c0d9f4a45ebafffcfe6cd9594981633b30b7 IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.0016394
0x8ae4a8d7b92b7b315bef0798e118171a6b1e208e69f98f9d695f120052245f91Approve556300162023-01-26 15:03:261 day 13 hrs ago0xb5155866a7f37edd924f9b3506b331651223580c IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00003576
0x6cadb6714b83b91f41d8a6998c5a278e8c1a454c0895b8c74e239356f184e52dApprove556297202023-01-26 15:01:501 day 13 hrs ago0x3e20d51e8943aa31882a094b773c1554c5a25bef IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00094845
0x275567ef3dd74074111a2148354041ec47a148dc6b3b0f888f73ecc51689136eExchange_underly...556041372023-01-26 12:47:471 day 15 hrs ago0x008b2bb61a01b4a91f17b4b745331fe545a8f915 IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00006428
0x45e3dd57669bb9036e4df4d26a0794a88759b406c9db8c239cb81e4504037767Approve554193042023-01-25 18:25:142 days 10 hrs ago0x6a61ea7832f84c3096c70f042ab88d9a56732d7b IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00165244
0x453bd8cfaa678bf9d7580b1d02e42ab9aa221174fa50db5905f7d4ad31fc74d9Approve553701442023-01-25 13:40:482 days 14 hrs ago0x3d7ebe72e6c787434d98846516bb9ab7fa160ebf IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00003062
0xddd6e6c7273296002ab40e8dee3ee96f7fa283d9e0a8ca142ce835a0c5f52de9Approve553700412023-01-25 13:40:132 days 14 hrs ago0x3d7ebe72e6c787434d98846516bb9ab7fa160ebf IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00003261
0x762204808ec7077549a3f91f43c2fed0a7d56a9e55b10940bbc11efa8abe0de6Approve553655682023-01-25 13:16:012 days 15 hrs ago0x6a61b6086695d9dd9663e786b8b66aa641c9a432 IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00084542
0xf4d52e2ddf6548cd52f7b8acf778d0435992adaf44a3cbc18205063caba2664fApprove553199682023-01-25 8:42:272 days 19 hrs ago0x49bdbbc511ec6bbc5d4cf0f45bbbfdd83470a64f IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00002906
0xfcaa6d3368ce1d10a39907d1f328f843757d00b422f5b36766649c0fae4448ccExchange_underly...553181972023-01-25 8:29:132 days 20 hrs ago0xf0c47cb3de395006330ec9387da98b13d20363a6 IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00005551
0x96105d1ccecab4245d7487454436d798f969abdb990c16ff996b07ef6a9e1b0fExchange_underly...552768622023-01-25 3:19:333 days 1 hr ago0x1164ce0bd345e28ac72ee6212ea4ecf560d88a4f IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00006765
0x75025492664da88cfc29b82146fcce6cb141eb70d58da541d2437bc8459954dfApprove551314762023-01-24 12:33:253 days 15 hrs ago0xbfc7e3604c3bb518a4a15f8ceeaf06ed48ac0de2 IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00002989
0x85f672b9cc6fdd650fceeb328210ede028bbcabd7651baef460d3d236f2ce619Approve551134032023-01-24 10:29:253 days 18 hrs ago0x31abb6131c35256b9af06d7e2037c22f8902d23d IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.0000329
0xb856ba5058eb978dec217cb63dad20f46d0fd0f93f24c9a018638d0e228c8f2cApprove551129852023-01-24 10:25:193 days 18 hrs ago0xbd035e63430e5a181941da38f5f1672f890dedf5 IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.0000323
0xcb08576af48ed1e6823ebd209de58eba349f0fa9d90c99056d9005c175d72277Approve550655642023-01-24 4:56:073 days 23 hrs ago0x478c9bef4702ca6758f3ed760a51677f74813df5 IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00085047
0xfa807d6bd1a980b6c5983c00b19d8b6a65f05f2f8b2ef85e9f1ce61e19d74e08Approve549938652023-01-23 20:10:424 days 8 hrs ago0xcbf511c3a56d2bf803ff026f17cad2c7bea0dba8 IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00005435
0xeccfd925d108e6f361816ab4b3d3975bbd4573c84bc9b7cb9adb41d58a8b03b1Approve549188672023-01-23 13:08:224 days 15 hrs ago0xc5b41fb105883bcd59ceef072b5b6bdbc5df31da IN 0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH0.00003068
[ Download CSV Export 
Latest 25 internal transaction
Parent Txn Hash Block From To Value
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac70x00000000000000000000000000000000000000040 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x7f90122bf0700f9e7e1f688fe926940e8839f3530 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac70x00000000000000000000000000000000000000040 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0xbf7e49483881c76487b0989cd7d9a8239b20ca410 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x09672362833d8f703d5395ef3252d4bfa51c15ca0 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x44407515a51cd8b69140d41d05b393bce8078b8a0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x09672362833d8f703d5395ef3252d4bfa51c15ca0 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x44407515a51cd8b69140d41d05b393bce8078b8a0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x44407515a51cd8b69140d41d05b393bce8078b8a0 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x7f90122bf0700f9e7e1f688fe926940e8839f3530 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x09672362833d8f703d5395ef3252d4bfa51c15ca0 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x7544fe3d184b6b55d6b36c3fca1157ee0ba302870x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x09672362833d8f703d5395ef3252d4bfa51c15ca0 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x44407515a51cd8b69140d41d05b393bce8078b8a0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x09672362833d8f703d5395ef3252d4bfa51c15ca0 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x44407515a51cd8b69140d41d05b393bce8078b8a0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x44407515a51cd8b69140d41d05b393bce8078b8a0 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x09672362833d8f703d5395ef3252d4bfa51c15ca0 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x44407515a51cd8b69140d41d05b393bce8078b8a0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x09672362833d8f703d5395ef3252d4bfa51c15ca0 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x44407515a51cd8b69140d41d05b393bce8078b8a0x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x44407515a51cd8b69140d41d05b393bce8078b8a0 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x09672362833d8f703d5395ef3252d4bfa51c15ca0 ETH
0x79c3a8e8659163d782a8369bd1d0c35db302b4ba3bcb1159ce5b3cbbfac2f091560244192023-01-28 3:23:061 hr 10 mins ago 0x7544fe3d184b6b55d6b36c3fca1157ee0ba302870x30df229cefa463e991e29d42db0bae2e122b2ac70 ETH
0xdfcb8dae1de4324f9b1859c63840163e6610c0805aef893eac02e076c40653c9560222672023-01-28 3:11:131 hr 22 mins ago 0x30df229cefa463e991e29d42db0bae2e122b2ac7 0x09672362833d8f703d5395ef3252d4bfa51c15ca0 ETH
[ Download CSV Export 
Loading

Minimal Proxy Contract for 0x09672362833d8f703d5395ef3252d4bfa51c15ca

Contract Name:
Vyper_contract

Compiler Version
vyper:0.2.16

Optimization Enabled:
N/A

Other Settings:
, None license
Decompile ByteCode

Contract Source Code (Vyper language format)

# @version 0.2.16
"""
@title StableSwap
@author Curve.Fi
@license Copyright (c) Curve.Fi, 2020-2021 - all rights reserved
@notice 3pool metapool implementation contract
@dev ERC20 support for return True/revert, return True/False, return None
"""

interface ERC20:
    def approve(_spender: address, _amount: uint256): nonpayable
    def balanceOf(_owner: address) -> uint256: view

interface Curve:
    def coins(i: uint256) -> address: view
    def get_virtual_price() -> uint256: view
    def calc_token_amount(amounts: uint256[BASE_N_COINS], deposit: bool) -> uint256: view
    def calc_withdraw_one_coin(_token_amount: uint256, i: int128) -> uint256: view
    def fee() -> uint256: view
    def get_dy(i: int128, j: int128, dx: uint256) -> uint256: view
    def exchange(i: int128, j: int128, dx: uint256, min_dy: uint256): nonpayable
    def add_liquidity(amounts: uint256[BASE_N_COINS], min_mint_amount: uint256): nonpayable
    def remove_liquidity_one_coin(_token_amount: uint256, i: int128, min_amount: uint256): nonpayable

interface Gauge:
    def set_rewards_receiver(_receiver: address): nonpayable
    def deposit(_value: uint256): nonpayable
    def withdraw(_value: uint256): nonpayable

interface GaugeExtension:
    def initialize(_base_gauge: address): nonpayable
    def checkpoint_rewards(_addr: address): nonpayable

interface Factory:
    def convert_metapool_fees() -> bool: nonpayable
    def get_fee_receiver(_pool: address) -> address: view
    def admin() -> address: view


event Transfer:
    sender: indexed(address)
    receiver: indexed(address)
    value: uint256

event Approval:
    owner: indexed(address)
    spender: indexed(address)
    value: uint256

event TokenExchange:
    buyer: indexed(address)
    sold_id: int128
    tokens_sold: uint256
    bought_id: int128
    tokens_bought: uint256

event TokenExchangeUnderlying:
    buyer: indexed(address)
    sold_id: int128
    tokens_sold: uint256
    bought_id: int128
    tokens_bought: uint256

event AddLiquidity:
    provider: indexed(address)
    token_amounts: uint256[N_COINS]
    fees: uint256[N_COINS]
    invariant: uint256
    token_supply: uint256

event RemoveLiquidity:
    provider: indexed(address)
    token_amounts: uint256[N_COINS]
    fees: uint256[N_COINS]
    token_supply: uint256

event RemoveLiquidityOne:
    provider: indexed(address)
    token_amount: uint256
    coin_amount: uint256
    token_supply: uint256

event RemoveLiquidityImbalance:
    provider: indexed(address)
    token_amounts: uint256[N_COINS]
    fees: uint256[N_COINS]
    invariant: uint256
    token_supply: uint256

event RampA:
    old_A: uint256
    new_A: uint256
    initial_time: uint256
    future_time: uint256

event StopRampA:
    A: uint256
    t: uint256


BASE_POOL: constant(address) = 0x7f90122BF0700F9E7e1F688fe926940E8839F353
BASE_N_COINS: constant(int128) = 2
BASE_COINS: constant(address[BASE_N_COINS]) = [0xFF970A61A04b1cA14834A43f5dE4533eBDDB5CC8, 0xFd086bC7CD5C481DCC9C85ebE478A1C0b69FCbb9]

BASE_LP_TOKEN: constant(address) = 0x7f90122BF0700F9E7e1F688fe926940E8839F353
BASE_GAUGE: constant(address) = 0xbF7E49483881C76487b0989CD7d9A8239B20CA41

GAUGE_EXTENSION_IMPL: constant(address) = 0x64448B78561690B70E17CBE8029a3e5c1bB7136e

N_COINS: constant(int128) = 2
MAX_COIN: constant(int128) = N_COINS - 1
PRECISION: constant(uint256) = 10 ** 18

FEE_DENOMINATOR: constant(uint256) = 10 ** 10
ADMIN_FEE: constant(uint256) = 5000000000

A_PRECISION: constant(uint256) = 100
MAX_A: constant(uint256) = 10 ** 6
MAX_A_CHANGE: constant(uint256) = 10
MIN_RAMP_TIME: constant(uint256) = 86400

factory: address

coins: public(address[N_COINS])
balances: public(uint256[N_COINS])
fee: public(uint256)  # fee * 1e10

initial_A: public(uint256)
future_A: public(uint256)
initial_A_time: public(uint256)
future_A_time: public(uint256)

rate_multiplier: uint256

name: public(String[64])
symbol: public(String[32])

balanceOf: public(HashMap[address, uint256])
allowance: public(HashMap[address, HashMap[address, uint256]])
totalSupply: public(uint256)

rewards_receiver: public(address)


@external
def __init__():
    # we do this to prevent the implementation contract from being used as a pool
    self.fee = 31337


@external
def initialize(
    _name: String[32],
    _symbol: String[10],
    _coin: address,
    _rate_multiplier: uint256,
    _A: uint256,
    _fee: uint256
):
    """
    @notice Contract initializer
    @param _name Name of the new pool
    @param _symbol Token symbol
    @param _coin Addresses of ERC20 conracts of coins
    @param _rate_multiplier Rate multiplier for `_coin` (10 ** (36 - decimals))
    @param _A Amplification coefficient multiplied by n ** (n - 1)
    @param _fee Fee to charge for exchanges
    """
    # check if fee was already set to prevent initializing contract twice
    assert self.fee == 0

    A: uint256 = _A * A_PRECISION
    self.coins = [_coin, BASE_LP_TOKEN]
    self.rate_multiplier = _rate_multiplier
    self.initial_A = A
    self.future_A = A
    self.fee = _fee
    self.factory = msg.sender

    self.name = concat("Curve.fi Factory USD Metapool: ", _name)
    self.symbol = concat(_symbol, "3CRV-f")

    receiver: address = create_forwarder_to(GAUGE_EXTENSION_IMPL)
    GaugeExtension(receiver).initialize(BASE_GAUGE)
    
    self.rewards_receiver = receiver

    Gauge(BASE_GAUGE).set_rewards_receiver(receiver)

    for coin in BASE_COINS:
        ERC20(coin).approve(BASE_POOL, MAX_UINT256)
    
    ERC20(BASE_LP_TOKEN).approve(BASE_GAUGE, MAX_UINT256)

    # fire a transfer event so block explorers identify the contract as an ERC20
    log Transfer(ZERO_ADDRESS, self, 0)


### ERC20 Functionality ###

@view
@external
def decimals() -> uint256:
    """
    @notice Get the number of decimals for this token
    @dev Implemented as a view method to reduce gas costs
    @return uint256 decimal places
    """
    return 18


@internal
def _transfer(_from: address, _to: address, _value: uint256):
    # # NOTE: vyper does not allow underflows
    # #       so the following subtraction would revert on insufficient balance
    receiver: address = self.rewards_receiver
    
    GaugeExtension(receiver).checkpoint_rewards(_from)
    self.balanceOf[_from] -= _value

    GaugeExtension(receiver).checkpoint_rewards(_to)
    self.balanceOf[_to] += _value

    log Transfer(_from, _to, _value)


@external
def transfer(_to : address, _value : uint256) -> bool:
    """
    @dev Transfer token for a specified address
    @param _to The address to transfer to.
    @param _value The amount to be transferred.
    """
    self._transfer(msg.sender, _to, _value)
    return True


@external
def transferFrom(_from : address, _to : address, _value : uint256) -> bool:
    """
     @dev Transfer tokens from one address to another.
     @param _from address The address which you want to send tokens from
     @param _to address The address which you want to transfer to
     @param _value uint256 the amount of tokens to be transferred
    """
    self._transfer(_from, _to, _value)

    _allowance: uint256 = self.allowance[_from][msg.sender]
    if _allowance != MAX_UINT256:
        self.allowance[_from][msg.sender] = _allowance - _value

    return True


@external
def approve(_spender : address, _value : uint256) -> bool:
    """
    @notice Approve the passed address to transfer the specified amount of
            tokens on behalf of msg.sender
    @dev Beware that changing an allowance via this method brings the risk that
         someone may use both the old and new allowance by unfortunate transaction
         ordering: https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
    @param _spender The address which will transfer the funds
    @param _value The amount of tokens that may be transferred
    @return bool success
    """
    self.allowance[msg.sender][_spender] = _value

    log Approval(msg.sender, _spender, _value)
    return True


### StableSwap Functionality ###

@view
@internal
def _A() -> uint256:
    """
    Handle ramping A up or down
    """
    t1: uint256 = self.future_A_time
    A1: uint256 = self.future_A

    if block.timestamp < t1:
        A0: uint256 = self.initial_A
        t0: uint256 = self.initial_A_time
        # Expressions in uint256 cannot have negative numbers, thus "if"
        if A1 > A0:
            return A0 + (A1 - A0) * (block.timestamp - t0) / (t1 - t0)
        else:
            return A0 - (A0 - A1) * (block.timestamp - t0) / (t1 - t0)

    else:  # when t1 == 0 or block.timestamp >= t1
        return A1


@view
@external
def admin_fee() -> uint256:
    return ADMIN_FEE


@view
@external
def A() -> uint256:
    return self._A() / A_PRECISION


@view
@external
def A_precise() -> uint256:
    return self._A()


@pure
@internal
def _xp_mem(_rates: uint256[N_COINS], _balances: uint256[N_COINS]) -> uint256[N_COINS]:
    result: uint256[N_COINS] = empty(uint256[N_COINS])
    for i in range(N_COINS):
        result[i] = _rates[i] * _balances[i] / PRECISION
    return result


@pure
@internal
def get_D(_xp: uint256[N_COINS], _amp: uint256) -> uint256:
    """
    D invariant calculation in non-overflowing integer operations
    iteratively

    A * sum(x_i) * n**n + D = A * D * n**n + D**(n+1) / (n**n * prod(x_i))

    Converging solution:
    D[j+1] = (A * n**n * sum(x_i) - D[j]**(n+1) / (n**n prod(x_i))) / (A * n**n - 1)
    """
    S: uint256 = 0
    Dprev: uint256 = 0
    for x in _xp:
        S += x
    if S == 0:
        return 0

    D: uint256 = S
    Ann: uint256 = _amp * N_COINS
    for i in range(255):
        D_P: uint256 = D
        for x in _xp:
            D_P = D_P * D / (x * N_COINS)  # If division by 0, this will be borked: only withdrawal will work. And that is good
        Dprev = D
        D = (Ann * S / A_PRECISION + D_P * N_COINS) * D / ((Ann - A_PRECISION) * D / A_PRECISION + (N_COINS + 1) * D_P)
        # Equality with the precision of 1
        if D > Dprev:
            if D - Dprev <= 1:
                return D
        else:
            if Dprev - D <= 1:
                return D
    # convergence typically occurs in 4 rounds or less, this should be unreachable!
    # if it does happen the pool is borked and LPs can withdraw via `remove_liquidity`
    raise


@view
@internal
def get_D_mem(_rates: uint256[N_COINS], _balances: uint256[N_COINS], _amp: uint256) -> uint256:
    xp: uint256[N_COINS] = self._xp_mem(_rates, _balances)
    return self.get_D(xp, _amp)


@view
@external
def get_virtual_price() -> uint256:
    """
    @notice The current virtual price of the pool LP token
    @dev Useful for calculating profits
    @return LP token virtual price normalized to 1e18
    """
    amp: uint256 = self._A()
    rates: uint256[N_COINS] = [self.rate_multiplier, Curve(BASE_POOL).get_virtual_price()]
    xp: uint256[N_COINS] = self._xp_mem(rates, self.balances)
    D: uint256 = self.get_D(xp, amp)
    # D is in the units similar to DAI (e.g. converted to precision 1e18)
    # When balanced, D = n * x_u - total virtual value of the portfolio
    return D * PRECISION / self.totalSupply


@view
@external
def calc_token_amount(_amounts: uint256[N_COINS], _is_deposit: bool) -> uint256:
    """
    @notice Calculate addition or reduction in token supply from a deposit or withdrawal
    @dev This calculation accounts for slippage, but not fees.
         Needed to prevent front-running, not for precise calculations!
    @param _amounts Amount of each coin being deposited
    @param _is_deposit set True for deposits, False for withdrawals
    @return Expected amount of LP tokens received
    """
    amp: uint256 = self._A()
    rates: uint256[N_COINS] = [self.rate_multiplier, Curve(BASE_POOL).get_virtual_price()]
    balances: uint256[N_COINS] = self.balances

    D0: uint256 = self.get_D_mem(rates, balances, amp)
    for i in range(N_COINS):
        amount: uint256 = _amounts[i]
        if _is_deposit:
            balances[i] += amount
        else:
            balances[i] -= amount
    D1: uint256 = self.get_D_mem(rates, balances, amp)
    diff: uint256 = 0
    if _is_deposit:
        diff = D1 - D0
    else:
        diff = D0 - D1
    return diff * self.totalSupply / D0


@external
@nonreentrant('lock')
def add_liquidity(
    _amounts: uint256[N_COINS],
    _min_mint_amount: uint256,
    _receiver: address = msg.sender
) -> uint256:
    """
    @notice Deposit coins into the pool
    @param _amounts List of amounts of coins to deposit
    @param _min_mint_amount Minimum amount of LP tokens to mint from the deposit
    @param _receiver Address that owns the minted LP tokens
    @return Amount of LP tokens received by depositing
    """
    amp: uint256 = self._A()
    rates: uint256[N_COINS] = [self.rate_multiplier, Curve(BASE_POOL).get_virtual_price()]

    # Initial invariant
    old_balances: uint256[N_COINS] = self.balances
    D0: uint256 = self.get_D_mem(rates, old_balances, amp)
    new_balances: uint256[N_COINS] = old_balances

    total_supply: uint256 = self.totalSupply
    for i in range(N_COINS):
        amount: uint256 = _amounts[i]
        if amount == 0:
            assert total_supply > 0
        else:
            response: Bytes[32] = raw_call(
                self.coins[i],
                _abi_encode(
                    msg.sender,
                    self,
                    amount,
                    method_id=method_id("transferFrom(address,address,uint256)")
                ),
                max_outsize=32,
            )
            if len(response) > 0:
                assert convert(response, bool)
            new_balances[i] += amount

    # Invariant after change
    D1: uint256 = self.get_D_mem(rates, new_balances, amp)
    assert D1 > D0

    # We need to recalculate the invariant accounting for fees
    # to calculate fair user's share
    fees: uint256[N_COINS] = empty(uint256[N_COINS])
    mint_amount: uint256 = 0
    if total_supply > 0:
        # Only account for fees if we are not the first to deposit
        base_fee: uint256 = self.fee * N_COINS / (4 * (N_COINS - 1))
        for i in range(N_COINS):
            ideal_balance: uint256 = D1 * old_balances[i] / D0
            difference: uint256 = 0
            new_balance: uint256 = new_balances[i]
            if ideal_balance > new_balance:
                difference = ideal_balance - new_balance
            else:
                difference = new_balance - ideal_balance
            fees[i] = base_fee * difference / FEE_DENOMINATOR
            self.balances[i] = new_balance - (fees[i] * ADMIN_FEE / FEE_DENOMINATOR)
            new_balances[i] -= fees[i]
        D2: uint256 = self.get_D_mem(rates, new_balances, amp)
        mint_amount = total_supply * (D2 - D0) / D0
    else:
        self.balances = new_balances
        mint_amount = D1  # Take the dust if there was any

    assert mint_amount >= _min_mint_amount

    # Deposit into base gauge
    # RewardsOnlyGauge and Factory LiquidityGauge allow
    # calling deposit with a value of 0
    Gauge(BASE_GAUGE).deposit(_amounts[MAX_COIN])

    GaugeExtension(self.rewards_receiver).checkpoint_rewards(_receiver)

    # Mint pool tokens
    total_supply += mint_amount
    self.balanceOf[_receiver] += mint_amount
    self.totalSupply = total_supply
    log Transfer(ZERO_ADDRESS, _receiver, mint_amount)
    log AddLiquidity(msg.sender, _amounts, fees, D1, total_supply)

    return mint_amount


@view
@internal
def get_y(i: int128, j: int128, x: uint256, xp: uint256[N_COINS]) -> uint256:
    # x in the input is converted to the same price/precision

    assert i != j       # dev: same coin
    assert j >= 0       # dev: j below zero
    assert j < N_COINS  # dev: j above N_COINS

    # should be unreachable, but good for safety
    assert i >= 0
    assert i < N_COINS

    amp: uint256 = self._A()
    D: uint256 = self.get_D(xp, amp)
    S_: uint256 = 0
    _x: uint256 = 0
    y_prev: uint256 = 0
    c: uint256 = D
    Ann: uint256 = amp * N_COINS

    for _i in range(N_COINS):
        if _i == i:
            _x = x
        elif _i != j:
            _x = xp[_i]
        else:
            continue
        S_ += _x
        c = c * D / (_x * N_COINS)

    c = c * D * A_PRECISION / (Ann * N_COINS)
    b: uint256 = S_ + D * A_PRECISION / Ann  # - D
    y: uint256 = D

    for _i in range(255):
        y_prev = y
        y = (y*y + c) / (2 * y + b - D)
        # Equality with the precision of 1
        if y > y_prev:
            if y - y_prev <= 1:
                return y
        else:
            if y_prev - y <= 1:
                return y
    raise


@view
@external
def get_dy(i: int128, j: int128, dx: uint256) -> uint256:
    """
    @notice Calculate the current output dy given input dx
    @dev Index values can be found via the `coins` public getter method
    @param i Index value for the coin to send
    @param j Index valie of the coin to recieve
    @param dx Amount of `i` being exchanged
    @return Amount of `j` predicted
    """
    rates: uint256[N_COINS] = [self.rate_multiplier, Curve(BASE_POOL).get_virtual_price()]
    xp: uint256[N_COINS] = self._xp_mem(rates, self.balances)

    x: uint256 = xp[i] + (dx * rates[i] / PRECISION)
    y: uint256 = self.get_y(i, j, x, xp)
    dy: uint256 = xp[j] - y - 1
    fee: uint256 = self.fee * dy / FEE_DENOMINATOR
    return (dy - fee) * PRECISION / rates[j]


@view
@external
def get_dy_underlying(i: int128, j: int128, dx: uint256) -> uint256:
    """
    @notice Calculate the current output dy given input dx on underlying
    @dev Index values can be found via the `coins` public getter method
    @param i Index value for the coin to send
    @param j Index valie of the coin to recieve
    @param dx Amount of `i` being exchanged
    @return Amount of `j` predicted
    """
    rates: uint256[N_COINS] = [self.rate_multiplier, Curve(BASE_POOL).get_virtual_price()]
    xp: uint256[N_COINS] = self._xp_mem(rates, self.balances)

    x: uint256 = 0
    base_i: int128 = 0
    base_j: int128 = 0
    meta_i: int128 = 0
    meta_j: int128 = 0

    if i != 0:
        base_i = i - MAX_COIN
        meta_i = 1
    if j != 0:
        base_j = j - MAX_COIN
        meta_j = 1

    if i == 0:
        x = xp[i] + dx * (rates[0] / 10**18)
    else:
        if j == 0:
            # i is from BasePool
            # At first, get the amount of pool tokens
            base_inputs: uint256[BASE_N_COINS] = empty(uint256[BASE_N_COINS])
            base_inputs[base_i] = dx
            # Token amount transformed to underlying "dollars"
            x = Curve(BASE_POOL).calc_token_amount(base_inputs, True) * rates[1] / PRECISION
            # Accounting for deposit/withdraw fees approximately
            x -= x * Curve(BASE_POOL).fee() / (2 * FEE_DENOMINATOR)
            # Adding number of pool tokens
            x += xp[MAX_COIN]
        else:
            # If both are from the base pool
            return Curve(BASE_POOL).get_dy(base_i, base_j, dx)

    # This pool is involved only when in-pool assets are used
    y: uint256 = self.get_y(meta_i, meta_j, x, xp)
    dy: uint256 = xp[meta_j] - y - 1
    dy = (dy - self.fee * dy / FEE_DENOMINATOR)

    # If output is going via the metapool
    if j == 0:
        dy /= (rates[0] / 10**18)
    else:
        # j is from BasePool
        # The fee is already accounted for
        dy = Curve(BASE_POOL).calc_withdraw_one_coin(dy * PRECISION / rates[1], base_j)

    return dy


@external
@nonreentrant('lock')
def exchange(
    i: int128,
    j: int128,
    _dx: uint256,
    _min_dy: uint256,
    _receiver: address = msg.sender,
) -> uint256:
    """
    @notice Perform an exchange between two coins
    @dev Index values can be found via the `coins` public getter method
    @param i Index value for the coin to send
    @param j Index valie of the coin to recieve
    @param _dx Amount of `i` being exchanged
    @param _min_dy Minimum amount of `j` to receive
    @param _receiver Address that receives `j`
    @return Actual amount of `j` received
    """
    rates: uint256[N_COINS] = [self.rate_multiplier, Curve(BASE_POOL).get_virtual_price()]

    old_balances: uint256[N_COINS] = self.balances
    xp: uint256[N_COINS] = self._xp_mem(rates, old_balances)

    x: uint256 = xp[i] + _dx * rates[i] / PRECISION
    y: uint256 = self.get_y(i, j, x, xp)

    dy: uint256 = xp[j] - y - 1  # -1 just in case there were some rounding errors
    dy_fee: uint256 = dy * self.fee / FEE_DENOMINATOR

    # Convert all to real units
    dy = (dy - dy_fee) * PRECISION / rates[j]
    assert dy >= _min_dy

    dy_admin_fee: uint256 = dy_fee * ADMIN_FEE / FEE_DENOMINATOR
    dy_admin_fee = dy_admin_fee * PRECISION / rates[j]

    # Change balances exactly in same way as we change actual ERC20 coin amounts
    self.balances[i] = old_balances[i] + _dx
    # When rounding errors happen, we undercharge admin fee in favor of LP
    self.balances[j] = old_balances[j] - dy - dy_admin_fee

    response: Bytes[32] = raw_call(
        self.coins[i],
        _abi_encode(
            msg.sender, self, _dx, method_id=method_id("transferFrom(address,address,uint256)")
        ),
        max_outsize=32,
    )
    if len(response) > 0:
        assert convert(response, bool)

    if i == MAX_COIN:
        Gauge(BASE_GAUGE).deposit(_dx)
    else:
        Gauge(BASE_GAUGE).withdraw(dy)

    response = raw_call(
        self.coins[j],
        _abi_encode(_receiver, dy, method_id=method_id("transfer(address,uint256)")),
        max_outsize=32,
    )
    if len(response) > 0:
        assert convert(response, bool)

    log TokenExchange(msg.sender, i, _dx, j, dy)

    return dy


@external
@nonreentrant('lock')
def exchange_underlying(
    i: int128,
    j: int128,
    _dx: uint256,
    _min_dy: uint256,
    _receiver: address = msg.sender,
) -> uint256:
    """
    @notice Perform an exchange between two underlying coins
    @param i Index value for the underlying coin to send
    @param j Index valie of the underlying coin to receive
    @param _dx Amount of `i` being exchanged
    @param _min_dy Minimum amount of `j` to receive
    @param _receiver Address that receives `j`
    @return Actual amount of `j` received
    """
    rates: uint256[N_COINS] = [self.rate_multiplier, Curve(BASE_POOL).get_virtual_price()]
    old_balances: uint256[N_COINS] = self.balances
    xp: uint256[N_COINS] = self._xp_mem(rates, old_balances)

    base_coins: address[BASE_N_COINS] = BASE_COINS

    dy: uint256 = 0
    base_i: int128 = 0
    base_j: int128 = 0
    meta_i: int128 = 0
    meta_j: int128 = 0
    x: uint256 = 0
    input_coin: address = ZERO_ADDRESS
    output_coin: address = ZERO_ADDRESS

    if i == 0:
        input_coin = self.coins[0]
    else:
        base_i = i - MAX_COIN
        meta_i = 1
        input_coin = base_coins[base_i]
    if j == 0:
        output_coin = self.coins[0]
    else:
        base_j = j - MAX_COIN
        meta_j = 1
        output_coin = base_coins[base_j]

    response: Bytes[32] = raw_call(
        input_coin,
        _abi_encode(
            msg.sender, self, _dx, method_id=method_id("transferFrom(address,address,uint256)")
        ),
        max_outsize=32,
    )
    if len(response) > 0:
        assert convert(response, bool)

    dx: uint256 = _dx
    if i == 0 or j == 0:
        if i == 0:
            x = xp[i] + dx * rates[i] / PRECISION
        else:
            # i is from BasePool
            # At first, get the amount of pool tokens
            base_inputs: uint256[BASE_N_COINS] = empty(uint256[BASE_N_COINS])
            base_inputs[base_i] = dx
            coin_i: address = self.coins[MAX_COIN]
            # Deposit and measure delta
            x = ERC20(coin_i).balanceOf(self)
            Curve(BASE_POOL).add_liquidity(base_inputs, 0)
            # Need to convert pool token to "virtual" units using rates
            # dx is also different now
            dx = ERC20(coin_i).balanceOf(self) - x
            x = dx * rates[MAX_COIN] / PRECISION
            # Adding number of pool tokens
            x += xp[MAX_COIN]
            Gauge(BASE_GAUGE).deposit(dx)

        y: uint256 = self.get_y(meta_i, meta_j, x, xp)

        # Either a real coin or token
        dy = xp[meta_j] - y - 1  # -1 just in case there were some rounding errors
        dy_fee: uint256 = dy * self.fee / FEE_DENOMINATOR

        # Convert all to real units
        # Works for both pool coins and real coins
        dy = (dy - dy_fee) * PRECISION / rates[meta_j]

        dy_admin_fee: uint256 = dy_fee * ADMIN_FEE / FEE_DENOMINATOR
        dy_admin_fee = dy_admin_fee * PRECISION / rates[meta_j]

        # Change balances exactly in same way as we change actual ERC20 coin amounts
        self.balances[meta_i] = old_balances[meta_i] + dx
        # When rounding errors happen, we undercharge admin fee in favor of LP
        self.balances[meta_j] = old_balances[meta_j] - dy - dy_admin_fee

        # Withdraw from the base pool if needed
        if j > 0:
            out_amount: uint256 = ERC20(output_coin).balanceOf(self)
            Gauge(BASE_GAUGE).withdraw(dy)
            Curve(BASE_POOL).remove_liquidity_one_coin(dy, base_j, 0)
            dy = ERC20(output_coin).balanceOf(self) - out_amount

        assert dy >= _min_dy

    else:
        # If both are from the base pool
        dy = ERC20(output_coin).balanceOf(self)
        Curve(BASE_POOL).exchange(base_i, base_j, dx, _min_dy)
        dy = ERC20(output_coin).balanceOf(self) - dy

    response = raw_call(
        output_coin,
        _abi_encode(_receiver, dy, method_id=method_id("transfer(address,uint256)")),
        max_outsize=32,
    )
    if len(response) > 0:
        assert convert(response, bool)

    log TokenExchangeUnderlying(msg.sender, i, dx, j, dy)

    return dy


@external
@nonreentrant('lock')
def remove_liquidity(
    _burn_amount: uint256,
    _min_amounts: uint256[N_COINS],
    _receiver: address = msg.sender
) -> uint256[N_COINS]:
    """
    @notice Withdraw coins from the pool
    @dev Withdrawal amounts are based on current deposit ratios
    @param _burn_amount Quantity of LP tokens to burn in the withdrawal
    @param _min_amounts Minimum amounts of underlying coins to receive
    @param _receiver Address that receives the withdrawn coins
    @return List of amounts of coins that were withdrawn
    """
    total_supply: uint256 = self.totalSupply
    amounts: uint256[N_COINS] = empty(uint256[N_COINS])

    for i in range(N_COINS):
        old_balance: uint256 = self.balances[i]
        value: uint256 = old_balance * _burn_amount / total_supply
        assert value >= _min_amounts[i]
        self.balances[i] = old_balance - value
        amounts[i] = value
        if i == MAX_COIN:
            Gauge(BASE_GAUGE).withdraw(value)
        response: Bytes[32] = raw_call(
            self.coins[i],
            _abi_encode(_receiver, value, method_id=method_id("transfer(address,uint256)")),
            max_outsize=32,
        )
        if len(response) > 0:
            assert convert(response, bool)


    GaugeExtension(self.rewards_receiver).checkpoint_rewards(msg.sender)

    total_supply -= _burn_amount
    self.balanceOf[msg.sender] -= _burn_amount
    self.totalSupply = total_supply
    log Transfer(msg.sender, ZERO_ADDRESS, _burn_amount)

    log RemoveLiquidity(msg.sender, amounts, empty(uint256[N_COINS]), total_supply)

    return amounts


@external
@nonreentrant('lock')
def remove_liquidity_imbalance(
    _amounts: uint256[N_COINS],
    _max_burn_amount: uint256,
    _receiver: address = msg.sender
) -> uint256:
    """
    @notice Withdraw coins from the pool in an imbalanced amount
    @param _amounts List of amounts of underlying coins to withdraw
    @param _max_burn_amount Maximum amount of LP token to burn in the withdrawal
    @param _receiver Address that receives the withdrawn coins
    @return Actual amount of the LP token burned in the withdrawal
    """
    amp: uint256 = self._A()
    rates: uint256[N_COINS] = [self.rate_multiplier, Curve(BASE_POOL).get_virtual_price()]
    old_balances: uint256[N_COINS] = self.balances
    D0: uint256 = self.get_D_mem(rates, old_balances, amp)

    new_balances: uint256[N_COINS] = old_balances
    for i in range(N_COINS):
        amount: uint256 = _amounts[i]
        if i == MAX_COIN:
            Gauge(BASE_GAUGE).withdraw(amount)
        if amount != 0:
            new_balances[i] -= amount
            response: Bytes[32] = raw_call(
                self.coins[i],
                _abi_encode(_receiver, amount, method_id=method_id("transfer(address,uint256)")),
                max_outsize=32,
            )
            if len(response) > 0:
                assert convert(response, bool)
    D1: uint256 = self.get_D_mem(rates, new_balances, amp)

    fees: uint256[N_COINS] = empty(uint256[N_COINS])
    base_fee: uint256 = self.fee * N_COINS / (4 * (N_COINS - 1))
    for i in range(N_COINS):
        ideal_balance: uint256 = D1 * old_balances[i] / D0
        difference: uint256 = 0
        new_balance: uint256 = new_balances[i]
        if ideal_balance > new_balance:
            difference = ideal_balance - new_balance
        else:
            difference = new_balance - ideal_balance
        fees[i] = base_fee * difference / FEE_DENOMINATOR
        self.balances[i] = new_balance - (fees[i] * ADMIN_FEE / FEE_DENOMINATOR)
        new_balances[i] -= fees[i]
    D2: uint256 = self.get_D_mem(rates, new_balances, amp)

    total_supply: uint256 = self.totalSupply
    burn_amount: uint256 = ((D0 - D2) * total_supply / D0) + 1
    assert burn_amount > 1  # dev: zero tokens burned
    assert burn_amount <= _max_burn_amount

    GaugeExtension(self.rewards_receiver).checkpoint_rewards(msg.sender)

    total_supply -= burn_amount
    self.totalSupply = total_supply
    self.balanceOf[msg.sender] -= burn_amount
    log Transfer(msg.sender, ZERO_ADDRESS, burn_amount)
    log RemoveLiquidityImbalance(msg.sender, _amounts, fees, D1, total_supply)

    return burn_amount


@view
@internal
def get_y_D(A: uint256, i: int128, xp: uint256[N_COINS], D: uint256) -> uint256:
    """
    Calculate x[i] if one reduces D from being calculated for xp to D

    Done by solving quadratic equation iteratively.
    x_1**2 + x1 * (sum' - (A*n**n - 1) * D / (A * n**n)) = D ** (n + 1) / (n ** (2 * n) * prod' * A)
    x_1**2 + b*x_1 = c

    x_1 = (x_1**2 + c) / (2*x_1 + b)
    """
    # x in the input is converted to the same price/precision

    assert i >= 0  # dev: i below zero
    assert i < N_COINS  # dev: i above N_COINS

    S_: uint256 = 0
    _x: uint256 = 0
    y_prev: uint256 = 0
    c: uint256 = D
    Ann: uint256 = A * N_COINS

    for _i in range(N_COINS):
        if _i != i:
            _x = xp[_i]
        else:
            continue
        S_ += _x
        c = c * D / (_x * N_COINS)

    c = c * D * A_PRECISION / (Ann * N_COINS)
    b: uint256 = S_ + D * A_PRECISION / Ann
    y: uint256 = D

    for _i in range(255):
        y_prev = y
        y = (y*y + c) / (2 * y + b - D)
        # Equality with the precision of 1
        if y > y_prev:
            if y - y_prev <= 1:
                return y
        else:
            if y_prev - y <= 1:
                return y
    raise


@view
@internal
def _calc_withdraw_one_coin(_burn_amount: uint256, i: int128) -> uint256[2]:
    # First, need to calculate
    # * Get current D
    # * Solve Eqn against y_i for D - _token_amount
    amp: uint256 = self._A()
    rates: uint256[N_COINS] = [self.rate_multiplier, Curve(BASE_POOL).get_virtual_price()]
    xp: uint256[N_COINS] = self._xp_mem(rates, self.balances)
    D0: uint256 = self.get_D(xp, amp)

    total_supply: uint256 = self.totalSupply
    D1: uint256 = D0 - _burn_amount * D0 / total_supply
    new_y: uint256 = self.get_y_D(amp, i, xp, D1)

    base_fee: uint256 = self.fee * N_COINS / (4 * (N_COINS - 1))
    xp_reduced: uint256[N_COINS] = empty(uint256[N_COINS])

    for j in range(N_COINS):
        dx_expected: uint256 = 0
        xp_j: uint256 = xp[j]
        if j == i:
            dx_expected = xp_j * D1 / D0 - new_y
        else:
            dx_expected = xp_j - xp_j * D1 / D0
        xp_reduced[j] = xp_j - base_fee * dx_expected / FEE_DENOMINATOR

    dy: uint256 = xp_reduced[i] - self.get_y_D(amp, i, xp_reduced, D1)
    dy_0: uint256 = (xp[i] - new_y) * PRECISION / rates[i]  # w/o fees
    dy = (dy - 1) * PRECISION / rates[i]  # Withdraw less to account for rounding errors

    return [dy, dy_0 - dy]


@view
@external
def calc_withdraw_one_coin(_burn_amount: uint256, i: int128) -> uint256:
    """
    @notice Calculate the amount received when withdrawing a single coin
    @param _burn_amount Amount of LP tokens to burn in the withdrawal
    @param i Index value of the coin to withdraw
    @return Amount of coin received
    """
    return self._calc_withdraw_one_coin(_burn_amount, i)[0]


@external
@nonreentrant('lock')
def remove_liquidity_one_coin(
    _burn_amount: uint256,
    i: int128,
    _min_received: uint256,
    _receiver: address = msg.sender,
) -> uint256:
    """
    @notice Withdraw a single coin from the pool
    @param _burn_amount Amount of LP tokens to burn in the withdrawal
    @param i Index value of the coin to withdraw
    @param _min_received Minimum amount of coin to receive
    @param _receiver Address that receives the withdrawn coins
    @return Amount of coin received
    """
    dy: uint256[2] = self._calc_withdraw_one_coin(_burn_amount, i)
    assert dy[0] >= _min_received

    self.balances[i] -= (dy[0] + dy[1] * ADMIN_FEE / FEE_DENOMINATOR)

    GaugeExtension(self.rewards_receiver).checkpoint_rewards(msg.sender)

    total_supply: uint256 = self.totalSupply - _burn_amount
    self.totalSupply = total_supply
    self.balanceOf[msg.sender] -= _burn_amount
    log Transfer(msg.sender, ZERO_ADDRESS, _burn_amount)

    if i == MAX_COIN:
        Gauge(BASE_GAUGE).withdraw(dy[0])

    response: Bytes[32] = raw_call(
        self.coins[i],
        _abi_encode(_receiver, dy[0], method_id=method_id("transfer(address,uint256)")),
        max_outsize=32,
    )
    if len(response) > 0:
        assert convert(response, bool)

    log RemoveLiquidityOne(msg.sender, _burn_amount, dy[0], total_supply)

    return dy[0]


@external
def ramp_A(_future_A: uint256, _future_time: uint256):
    assert msg.sender == Factory(self.factory).admin()  # dev: only owner
    assert block.timestamp >= self.initial_A_time + MIN_RAMP_TIME
    assert _future_time >= block.timestamp + MIN_RAMP_TIME  # dev: insufficient time

    _initial_A: uint256 = self._A()
    _future_A_p: uint256 = _future_A * A_PRECISION

    assert _future_A > 0 and _future_A < MAX_A
    if _future_A_p < _initial_A:
        assert _future_A_p * MAX_A_CHANGE >= _initial_A
    else:
        assert _future_A_p <= _initial_A * MAX_A_CHANGE

    self.initial_A = _initial_A
    self.future_A = _future_A_p
    self.initial_A_time = block.timestamp
    self.future_A_time = _future_time

    log RampA(_initial_A, _future_A_p, block.timestamp, _future_time)


@external
def stop_ramp_A():
    assert msg.sender == Factory(self.factory).admin()  # dev: only owner

    current_A: uint256 = self._A()
    self.initial_A = current_A
    self.future_A = current_A
    self.initial_A_time = block.timestamp
    self.future_A_time = block.timestamp
    # now (block.timestamp < t1) is always False, so we return saved A

    log StopRampA(current_A, block.timestamp)


@view
@external
def admin_balances(i: uint256) -> uint256:
    coin: address = self.coins[i]
    if i == MAX_COIN:
        coin = BASE_GAUGE
    return ERC20(coin).balanceOf(self) - self.balances[i]


@external
def withdraw_admin_fees():
    # transfer coin 0 to Factory and call `convert_fees` to swap it for coin 1
    factory: address = self.factory
    coin: address = self.coins[0]
    amount: uint256 = ERC20(coin).balanceOf(self) - self.balances[0]
    if amount > 0:
        response: Bytes[32] = raw_call(
            coin,
            _abi_encode(factory, amount, method_id=method_id("transfer(address,uint256)")),
            max_outsize=32,
        )
        if len(response) > 0:
            assert convert(response, bool)
        Factory(factory).convert_metapool_fees()

    # transfer coin 1 to the receiver
    coin = self.coins[1]
    amount = ERC20(BASE_GAUGE).balanceOf(self) - self.balances[1]
    Gauge(BASE_GAUGE).withdraw(amount)
    amount = ERC20(coin).balanceOf(self)
    if amount > 0:
        receiver: address = Factory(factory).get_fee_receiver(self)
        response: Bytes[32] = raw_call(
            coin,
            _abi_encode(receiver, amount, method_id=method_id("transfer(address,uint256)")),
            max_outsize=32,
        )
        if len(response) > 0:
            assert convert(response, bool)

Contract ABI

[{"name":"Transfer","inputs":[{"name":"sender","type":"address","indexed":true},{"name":"receiver","type":"address","indexed":true},{"name":"value","type":"uint256","indexed":false}],"anonymous":false,"type":"event"},{"name":"Approval","inputs":[{"name":"owner","type":"address","indexed":true},{"name":"spender","type":"address","indexed":true},{"name":"value","type":"uint256","indexed":false}],"anonymous":false,"type":"event"},{"name":"TokenExchange","inputs":[{"name":"buyer","type":"address","indexed":true},{"name":"sold_id","type":"int128","indexed":false},{"name":"tokens_sold","type":"uint256","indexed":false},{"name":"bought_id","type":"int128","indexed":false},{"name":"tokens_bought","type":"uint256","indexed":false}],"anonymous":false,"type":"event"},{"name":"TokenExchangeUnderlying","inputs":[{"name":"buyer","type":"address","indexed":true},{"name":"sold_id","type":"int128","indexed":false},{"name":"tokens_sold","type":"uint256","indexed":false},{"name":"bought_id","type":"int128","indexed":false},{"name":"tokens_bought","type":"uint256","indexed":false}],"anonymous":false,"type":"event"},{"name":"AddLiquidity","inputs":[{"name":"provider","type":"address","indexed":true},{"name":"token_amounts","type":"uint256[2]","indexed":false},{"name":"fees","type":"uint256[2]","indexed":false},{"name":"invariant","type":"uint256","indexed":false},{"name":"token_supply","type":"uint256","indexed":false}],"anonymous":false,"type":"event"},{"name":"RemoveLiquidity","inputs":[{"name":"provider","type":"address","indexed":true},{"name":"token_amounts","type":"uint256[2]","indexed":false},{"name":"fees","type":"uint256[2]","indexed":false},{"name":"token_supply","type":"uint256","indexed":false}],"anonymous":false,"type":"event"},{"name":"RemoveLiquidityOne","inputs":[{"name":"provider","type":"address","indexed":true},{"name":"token_amount","type":"uint256","indexed":false},{"name":"coin_amount","type":"uint256","indexed":false},{"name":"token_supply","type":"uint256","indexed":false}],"anonymous":false,"type":"event"},{"name":"RemoveLiquidityImbalance","inputs":[{"name":"provider","type":"address","indexed":true},{"name":"token_amounts","type":"uint256[2]","indexed":false},{"name":"fees","type":"uint256[2]","indexed":false},{"name":"invariant","type":"uint256","indexed":false},{"name":"token_supply","type":"uint256","indexed":false}],"anonymous":false,"type":"event"},{"name":"RampA","inputs":[{"name":"old_A","type":"uint256","indexed":false},{"name":"new_A","type":"uint256","indexed":false},{"name":"initial_time","type":"uint256","indexed":false},{"name":"future_time","type":"uint256","indexed":false}],"anonymous":false,"type":"event"},{"name":"StopRampA","inputs":[{"name":"A","type":"uint256","indexed":false},{"name":"t","type":"uint256","indexed":false}],"anonymous":false,"type":"event"},{"stateMutability":"nonpayable","type":"constructor","inputs":[],"outputs":[]},{"stateMutability":"nonpayable","type":"function","name":"initialize","inputs":[{"name":"_name","type":"string"},{"name":"_symbol","type":"string"},{"name":"_coin","type":"address"},{"name":"_rate_multiplier","type":"uint256"},{"name":"_A","type":"uint256"},{"name":"_fee","type":"uint256"}],"outputs":[],"gas":539992},{"stateMutability":"view","type":"function","name":"decimals","inputs":[],"outputs":[{"name":"","type":"uint256"}],"gas":318},{"stateMutability":"nonpayable","type":"function","name":"transfer","inputs":[{"name":"_to","type":"address"},{"name":"_value","type":"uint256"}],"outputs":[{"name":"","type":"bool"}],"gas":91298},{"stateMutability":"nonpayable","type":"function","name":"transferFrom","inputs":[{"name":"_from","type":"address"},{"name":"_to","type":"address"},{"name":"_value","type":"uint256"}],"outputs":[{"name":"","type":"bool"}],"gas":129233},{"stateMutability":"nonpayable","type":"function","name":"approve","inputs":[{"name":"_spender","type":"address"},{"name":"_value","type":"uint256"}],"outputs":[{"name":"","type":"bool"}],"gas":39151},{"stateMutability":"view","type":"function","name":"admin_fee","inputs":[],"outputs":[{"name":"","type":"uint256"}],"gas":438},{"stateMutability":"view","type":"function","name":"A","inputs":[],"outputs":[{"name":"","type":"uint256"}],"gas":10704},{"stateMutability":"view","type":"function","name":"A_precise","inputs":[],"outputs":[{"name":"","type":"uint256"}],"gas":10666},{"stateMutability":"view","type":"function","name":"get_virtual_price","inputs":[],"outputs":[{"name":"","type":"uint256"}],"gas":1023280},{"stateMutability":"view","type":"function","name":"calc_token_amount","inputs":[{"name":"_amounts","type":"uint256[2]"},{"name":"_is_deposit","type":"bool"}],"outputs":[{"name":"","type":"uint256"}],"gas":4029742},{"stateMutability":"nonpayable","type":"function","name":"add_liquidity","inputs":[{"name":"_amounts","type":"uint256[2]"},{"name":"_min_mint_amount","type":"uint256"}],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"nonpayable","type":"function","name":"add_liquidity","inputs":[{"name":"_amounts","type":"uint256[2]"},{"name":"_min_mint_amount","type":"uint256"},{"name":"_receiver","type":"address"}],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"view","type":"function","name":"get_dy","inputs":[{"name":"i","type":"int128"},{"name":"j","type":"int128"},{"name":"dx","type":"uint256"}],"outputs":[{"name":"","type":"uint256"}],"gas":2466434},{"stateMutability":"view","type":"function","name":"get_dy_underlying","inputs":[{"name":"i","type":"int128"},{"name":"j","type":"int128"},{"name":"dx","type":"uint256"}],"outputs":[{"name":"","type":"uint256"}],"gas":2474920},{"stateMutability":"nonpayable","type":"function","name":"exchange","inputs":[{"name":"i","type":"int128"},{"name":"j","type":"int128"},{"name":"_dx","type":"uint256"},{"name":"_min_dy","type":"uint256"}],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"nonpayable","type":"function","name":"exchange","inputs":[{"name":"i","type":"int128"},{"name":"j","type":"int128"},{"name":"_dx","type":"uint256"},{"name":"_min_dy","type":"uint256"},{"name":"_receiver","type":"address"}],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"nonpayable","type":"function","name":"exchange_underlying","inputs":[{"name":"i","type":"int128"},{"name":"j","type":"int128"},{"name":"_dx","type":"uint256"},{"name":"_min_dy","type":"uint256"}],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"nonpayable","type":"function","name":"exchange_underlying","inputs":[{"name":"i","type":"int128"},{"name":"j","type":"int128"},{"name":"_dx","type":"uint256"},{"name":"_min_dy","type":"uint256"},{"name":"_receiver","type":"address"}],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"nonpayable","type":"function","name":"remove_liquidity","inputs":[{"name":"_burn_amount","type":"uint256"},{"name":"_min_amounts","type":"uint256[2]"}],"outputs":[{"name":"","type":"uint256[2]"}]},{"stateMutability":"nonpayable","type":"function","name":"remove_liquidity","inputs":[{"name":"_burn_amount","type":"uint256"},{"name":"_min_amounts","type":"uint256[2]"},{"name":"_receiver","type":"address"}],"outputs":[{"name":"","type":"uint256[2]"}]},{"stateMutability":"nonpayable","type":"function","name":"remove_liquidity_imbalance","inputs":[{"name":"_amounts","type":"uint256[2]"},{"name":"_max_burn_amount","type":"uint256"}],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"nonpayable","type":"function","name":"remove_liquidity_imbalance","inputs":[{"name":"_amounts","type":"uint256[2]"},{"name":"_max_burn_amount","type":"uint256"},{"name":"_receiver","type":"address"}],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"view","type":"function","name":"calc_withdraw_one_coin","inputs":[{"name":"_burn_amount","type":"uint256"},{"name":"i","type":"int128"}],"outputs":[{"name":"","type":"uint256"}],"gas":1108},{"stateMutability":"nonpayable","type":"function","name":"remove_liquidity_one_coin","inputs":[{"name":"_burn_amount","type":"uint256"},{"name":"i","type":"int128"},{"name":"_min_received","type":"uint256"}],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"nonpayable","type":"function","name":"remove_liquidity_one_coin","inputs":[{"name":"_burn_amount","type":"uint256"},{"name":"i","type":"int128"},{"name":"_min_received","type":"uint256"},{"name":"_receiver","type":"address"}],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"nonpayable","type":"function","name":"ramp_A","inputs":[{"name":"_future_A","type":"uint256"},{"name":"_future_time","type":"uint256"}],"outputs":[],"gas":161967},{"stateMutability":"nonpayable","type":"function","name":"stop_ramp_A","inputs":[],"outputs":[],"gas":157887},{"stateMutability":"view","type":"function","name":"admin_balances","inputs":[{"name":"i","type":"uint256"}],"outputs":[{"name":"","type":"uint256"}],"gas":7855},{"stateMutability":"nonpayable","type":"function","name":"withdraw_admin_fees","inputs":[],"outputs":[],"gas":43650},{"stateMutability":"view","type":"function","name":"coins","inputs":[{"name":"arg0","type":"uint256"}],"outputs":[{"name":"","type":"address"}],"gas":3123},{"stateMutability":"view","type":"function","name":"balances","inputs":[{"name":"arg0","type":"uint256"}],"outputs":[{"name":"","type":"uint256"}],"gas":3153},{"stateMutability":"view","type":"function","name":"fee","inputs":[],"outputs":[{"name":"","type":"uint256"}],"gas":3138},{"stateMutability":"view","type":"function","name":"initial_A","inputs":[],"outputs":[{"name":"","type":"uint256"}],"gas":3168},{"stateMutability":"view","type":"function","name":"future_A","inputs":[],"outputs":[{"name":"","type":"uint256"}],"gas":3198},{"stateMutability":"view","type":"function","name":"initial_A_time","inputs":[],"outputs":[{"name":"","type":"uint256"}],"gas":3228},{"stateMutability":"view","type":"function","name":"future_A_time","inputs":[],"outputs":[{"name":"","type":"uint256"}],"gas":3258},{"stateMutability":"view","type":"function","name":"name","inputs":[],"outputs":[{"name":"","type":"string"}],"gas":13518},{"stateMutability":"view","type":"function","name":"symbol","inputs":[],"outputs":[{"name":"","type":"string"}],"gas":11271},{"stateMutability":"view","type":"function","name":"balanceOf","inputs":[{"name":"arg0","type":"address"}],"outputs":[{"name":"","type":"uint256"}],"gas":3563},{"stateMutability":"view","type":"function","name":"allowance","inputs":[{"name":"arg0","type":"address"},{"name":"arg1","type":"address"}],"outputs":[{"name":"","type":"uint256"}],"gas":3808},{"stateMutability":"view","type":"function","name":"totalSupply","inputs":[],"outputs":[{"name":"","type":"uint256"}],"gas":3408},{"stateMutability":"view","type":"function","name":"rewards_receiver","inputs":[],"outputs":[{"name":"","type":"address"}],"gas":3438}]

Block Transaction Gas Used Reward
Age Block Fee Address BC Fee Address Voting Power Jailed Incoming
Block Uncle Number Difficulty Gas Used Reward
Loading
Make sure to use the "Vote Down" button for any spammy posts, and the "Vote Up" for interesting conversations.