ETH Price: $2,542.72 (+0.58%)
Gas: 0.46 Gwei

Transaction Decoder

Block:
13753258 at Dec-06-2021 04:07:48 PM +UTC
Transaction Fee:
0.016855900232162616 ETH $42.86
Gas Used:
117,924 Gas / 142.938674334 Gwei

Emitted Events:

168 Vyper_contract.Transfer( _from=[Sender] 0xd8ed830e33859af668bf4bf1550dfe4cc9984157, _to=[Receiver] Vyper_contract, _value=215310000000000000000 )
169 Vyper_contract.Transfer( sender=0x0000000000000000000000000000000000000000, receiver=[Sender] 0xd8ed830e33859af668bf4bf1550dfe4cc9984157, value=215782115491421084556 )
170 Vyper_contract.AddLiquidity( provider=[Sender] 0xd8ed830e33859af668bf4bf1550dfe4cc9984157, token_amounts=[215310000000000000000, 0], fees=[107153490508454270, 108934242675281908], invariant=55219310568814415465840303, token_supply=54858810785626255969212194 )

Account State Difference:

  Address   Before After State Difference Code
(2Miners: PPLNS)
4,043.53309548738063042 Eth4,043.53327119414063042 Eth0.00017570676
0x9D046499...DEDf279e8
0xD533a949...bA034cd52
0xD8eD830E...CC9984157
0.344578741989108006 Eth
Nonce: 1094
0.32772284175694539 Eth
Nonce: 1095
0.016855900232162616

Execution Trace

Vyper_contract.add_liquidity( _amounts=[215310000000000000000, 0], _min_mint_amount=213676757456061864845 ) => ( 215782115491421084556 )
  • Vyper_contract.add_liquidity( _amounts=[215310000000000000000, 0], _min_mint_amount=213676757456061864845 ) => ( 215782115491421084556 )
    • Vyper_contract.transferFrom( _from=0xD8eD830E33859AF668Bf4bF1550DFe4CC9984157, _to=0x9D0464996170c6B9e75eED71c68B99dDEDf279e8, _value=215310000000000000000 ) => ( True )
      File 1 of 3: Vyper_contract
      # @version 0.2.15
      """
      @title StableSwap
      @author Curve.Fi
      @license Copyright (c) Curve.Fi, 2020-2021 - all rights reserved
      @notice 2 coin pool implementation with no lending
      @dev Optimized to only support ERC20's with 18 decimals that return True/revert
      """
      
      from vyper.interfaces import ERC20
      
      interface Factory:
          def convert_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 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
      
      
      N_COINS: constant(int128) = 2
      PRECISION: constant(int128) = 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)
      
      name: public(String[64])
      symbol: public(String[32])
      
      balanceOf: public(HashMap[address, uint256])
      allowance: public(HashMap[address, HashMap[address, uint256]])
      totalSupply: public(uint256)
      
      
      @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],
          _coins: address[4],
          _rate_multipliers: uint256[4],
          _A: uint256,
          _fee: uint256,
      ):
          """
          @notice Contract constructor
          @param _name Name of the new pool
          @param _symbol Token symbol
          @param _coins List of all ERC20 conract addresses of coins
          @param _rate_multipliers List of number of decimals in coins
          @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
      
          for i in range(N_COINS):
              coin: address = _coins[i]
              if coin == ZERO_ADDRESS:
                  break
              self.coins[i] = coin
              assert _rate_multipliers[i] == PRECISION
      
          A: uint256 = _A * A_PRECISION
          self.initial_A = A
          self.future_A = A
          self.fee = _fee
          self.factory = msg.sender
      
          self.name = concat("Curve.fi Factory Plain Pool: ", _name)
          self.symbol = concat(_symbol, "-f")
      
          # 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
          self.balanceOf[_from] -= _value
          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
      @external
      def get_balances() -> uint256[N_COINS]:
          return self.balances
      
      
      @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 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
          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 * D / _xp[0] * D / _xp[1] / (N_COINS)**2
              Dprev: uint256 = 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
      @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()
          D: uint256 = self.get_D(self.balances, 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()
          balances: uint256[N_COINS] = self.balances
      
          D0: uint256 = self.get_D(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(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()
          old_balances: uint256[N_COINS] = self.balances
      
          # Initial invariant
          D0: uint256 = self.get_D(old_balances, amp)
      
          total_supply: uint256 = self.totalSupply
          new_balances: uint256[N_COINS] = old_balances
          for i in range(N_COINS):
              amount: uint256 = _amounts[i]
              if total_supply == 0:
                  assert amount > 0  # dev: initial deposit requires all coins
              new_balances[i] += amount
      
          # Invariant after change
          D1: uint256 = self.get_D(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(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, "Slippage screwed you"
      
          # Take coins from the sender
          for i in range(N_COINS):
              amount: uint256 = _amounts[i]
              if amount > 0:
                  assert ERC20(self.coins[i]).transferFrom(msg.sender, self, amount)
      
          # 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:
          """
          Calculate x[j] if one makes x[i] = x
      
          Done by solving quadratic equation iteratively.
          x_1**2 + x_1 * (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 != 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
          """
          xp: uint256[N_COINS] = self.balances
      
          x: uint256 = xp[i] + dx
          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
      
      
      @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
          @return Actual amount of `j` received
          """
          old_balances: uint256[N_COINS] = self.balances
      
          x: uint256 = old_balances[i] + _dx
          y: uint256 = self.get_y(i, j, x, old_balances)
      
          dy: uint256 = old_balances[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_fee
          assert dy >= _min_dy, "Exchange resulted in fewer coins than expected"
      
          dy_admin_fee: uint256 = dy_fee * ADMIN_FEE / FEE_DENOMINATOR
      
          # 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
      
          assert ERC20(self.coins[i]).transferFrom(msg.sender, self, _dx)
          assert ERC20(self.coins[j]).transfer(_receiver, dy)
      
          log TokenExchange(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], "Withdrawal resulted in fewer coins than expected"
              self.balances[i] = old_balance - value
              amounts[i] = value
              assert ERC20(self.coins[i]).transfer(_receiver, value)
      
          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()
          old_balances: uint256[N_COINS] = self.balances
          D0: uint256 = self.get_D(old_balances, amp)
      
          new_balances: uint256[N_COINS] = old_balances
          for i in range(N_COINS):
              new_balances[i] -= _amounts[i]
          D1: uint256 = self.get_D(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(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, "Slippage screwed you"
      
          total_supply -= burn_amount
          self.totalSupply = total_supply
          self.balanceOf[msg.sender] -= burn_amount
          log Transfer(msg.sender, ZERO_ADDRESS, burn_amount)
      
          for i in range(N_COINS):
              if _amounts[i] != 0:
                  assert ERC20(self.coins[i]).transfer(_receiver, _amounts[i])
      
          log RemoveLiquidityImbalance(msg.sender, _amounts, fees, D1, total_supply)
      
          return burn_amount
      
      
      @pure
      @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 + x_1 * (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()
          balances: uint256[N_COINS] = self.balances
          D0: uint256 = self.get_D(balances, amp)
      
          total_supply: uint256 = self.totalSupply
          D1: uint256 = D0 - _burn_amount * D0 / total_supply
          new_y: uint256 = self.get_y_D(amp, i, balances, 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 = balances[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 = (balances[i] - new_y)  # w/o fees
          dy = (dy - 1)  # Withdraw less to account for rounding errors
      
          return [dy, dy_0 - dy]
      
      
      @view
      @external
      def calc_withdraw_one_coin(_burn_amount: uint256, i: int128, _previous: bool = False) -> 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, "Not enough coins removed"
      
          self.balances[i] -= (dy[0] + dy[1] * ADMIN_FEE / FEE_DENOMINATOR)
          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)
      
          assert ERC20(self.coins[i]).transfer(_receiver, dy[0])
      
          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:
          return ERC20(self.coins[i]).balanceOf(self) - self.balances[i]
      
      
      @external
      def withdraw_admin_fees():
          receiver: address = Factory(self.factory).get_fee_receiver(self)
      
          for i in range(N_COINS):
              coin: address = self.coins[i]
              fees: uint256 = ERC20(coin).balanceOf(self) - self.balances[i]
              ERC20(coin).transfer(receiver, fees)

      File 2 of 3: Vyper_contract
      # @version 0.2.4
      """
      @title Curve DAO Token
      @author Curve Finance
      @license MIT
      @notice ERC20 with piecewise-linear mining supply.
      @dev Based on the ERC-20 token standard as defined at
           https://eips.ethereum.org/EIPS/eip-20
      """
      
      from vyper.interfaces import ERC20
      
      implements: ERC20
      
      
      event Transfer:
          _from: indexed(address)
          _to: indexed(address)
          _value: uint256
      
      event Approval:
          _owner: indexed(address)
          _spender: indexed(address)
          _value: uint256
      
      event UpdateMiningParameters:
          time: uint256
          rate: uint256
          supply: uint256
      
      event SetMinter:
          minter: address
      
      event SetAdmin:
          admin: address
      
      
      name: public(String[64])
      symbol: public(String[32])
      decimals: public(uint256)
      
      balanceOf: public(HashMap[address, uint256])
      allowances: HashMap[address, HashMap[address, uint256]]
      total_supply: uint256
      
      minter: public(address)
      admin: public(address)
      
      # General constants
      YEAR: constant(uint256) = 86400 * 365
      
      # Allocation:
      # =========
      # * shareholders - 30%
      # * emplyees - 3%
      # * DAO-controlled reserve - 5%
      # * Early users - 5%
      # == 43% ==
      # left for inflation: 57%
      
      # Supply parameters
      INITIAL_SUPPLY: constant(uint256) = 1_303_030_303
      INITIAL_RATE: constant(uint256) = 274_815_283 * 10 ** 18 / YEAR  # leading to 43% premine
      RATE_REDUCTION_TIME: constant(uint256) = YEAR
      RATE_REDUCTION_COEFFICIENT: constant(uint256) = 1189207115002721024  # 2 ** (1/4) * 1e18
      RATE_DENOMINATOR: constant(uint256) = 10 ** 18
      INFLATION_DELAY: constant(uint256) = 86400
      
      # Supply variables
      mining_epoch: public(int128)
      start_epoch_time: public(uint256)
      rate: public(uint256)
      
      start_epoch_supply: uint256
      
      
      @external
      def __init__(_name: String[64], _symbol: String[32], _decimals: uint256):
          """
          @notice Contract constructor
          @param _name Token full name
          @param _symbol Token symbol
          @param _decimals Number of decimals for token
          """
          init_supply: uint256 = INITIAL_SUPPLY * 10 ** _decimals
          self.name = _name
          self.symbol = _symbol
          self.decimals = _decimals
          self.balanceOf[msg.sender] = init_supply
          self.total_supply = init_supply
          self.admin = msg.sender
          log Transfer(ZERO_ADDRESS, msg.sender, init_supply)
      
          self.start_epoch_time = block.timestamp + INFLATION_DELAY - RATE_REDUCTION_TIME
          self.mining_epoch = -1
          self.rate = 0
          self.start_epoch_supply = init_supply
      
      
      @internal
      def _update_mining_parameters():
          """
          @dev Update mining rate and supply at the start of the epoch
               Any modifying mining call must also call this
          """
          _rate: uint256 = self.rate
          _start_epoch_supply: uint256 = self.start_epoch_supply
      
          self.start_epoch_time += RATE_REDUCTION_TIME
          self.mining_epoch += 1
      
          if _rate == 0:
              _rate = INITIAL_RATE
          else:
              _start_epoch_supply += _rate * RATE_REDUCTION_TIME
              self.start_epoch_supply = _start_epoch_supply
              _rate = _rate * RATE_DENOMINATOR / RATE_REDUCTION_COEFFICIENT
      
          self.rate = _rate
      
          log UpdateMiningParameters(block.timestamp, _rate, _start_epoch_supply)
      
      
      @external
      def update_mining_parameters():
          """
          @notice Update mining rate and supply at the start of the epoch
          @dev Callable by any address, but only once per epoch
               Total supply becomes slightly larger if this function is called late
          """
          assert block.timestamp >= self.start_epoch_time + RATE_REDUCTION_TIME  # dev: too soon!
          self._update_mining_parameters()
      
      
      @external
      def start_epoch_time_write() -> uint256:
          """
          @notice Get timestamp of the current mining epoch start
                  while simultaneously updating mining parameters
          @return Timestamp of the epoch
          """
          _start_epoch_time: uint256 = self.start_epoch_time
          if block.timestamp >= _start_epoch_time + RATE_REDUCTION_TIME:
              self._update_mining_parameters()
              return self.start_epoch_time
          else:
              return _start_epoch_time
      
      
      @external
      def future_epoch_time_write() -> uint256:
          """
          @notice Get timestamp of the next mining epoch start
                  while simultaneously updating mining parameters
          @return Timestamp of the next epoch
          """
          _start_epoch_time: uint256 = self.start_epoch_time
          if block.timestamp >= _start_epoch_time + RATE_REDUCTION_TIME:
              self._update_mining_parameters()
              return self.start_epoch_time + RATE_REDUCTION_TIME
          else:
              return _start_epoch_time + RATE_REDUCTION_TIME
      
      
      @internal
      @view
      def _available_supply() -> uint256:
          return self.start_epoch_supply + (block.timestamp - self.start_epoch_time) * self.rate
      
      
      @external
      @view
      def available_supply() -> uint256:
          """
          @notice Current number of tokens in existence (claimed or unclaimed)
          """
          return self._available_supply()
      
      
      @external
      @view
      def mintable_in_timeframe(start: uint256, end: uint256) -> uint256:
          """
          @notice How much supply is mintable from start timestamp till end timestamp
          @param start Start of the time interval (timestamp)
          @param end End of the time interval (timestamp)
          @return Tokens mintable from `start` till `end`
          """
          assert start <= end  # dev: start > end
          to_mint: uint256 = 0
          current_epoch_time: uint256 = self.start_epoch_time
          current_rate: uint256 = self.rate
      
          # Special case if end is in future (not yet minted) epoch
          if end > current_epoch_time + RATE_REDUCTION_TIME:
              current_epoch_time += RATE_REDUCTION_TIME
              current_rate = current_rate * RATE_DENOMINATOR / RATE_REDUCTION_COEFFICIENT
      
          assert end <= current_epoch_time + RATE_REDUCTION_TIME  # dev: too far in future
      
          for i in range(999):  # Curve will not work in 1000 years. Darn!
              if end >= current_epoch_time:
                  current_end: uint256 = end
                  if current_end > current_epoch_time + RATE_REDUCTION_TIME:
                      current_end = current_epoch_time + RATE_REDUCTION_TIME
      
                  current_start: uint256 = start
                  if current_start >= current_epoch_time + RATE_REDUCTION_TIME:
                      break  # We should never get here but what if...
                  elif current_start < current_epoch_time:
                      current_start = current_epoch_time
      
                  to_mint += current_rate * (current_end - current_start)
      
                  if start >= current_epoch_time:
                      break
      
              current_epoch_time -= RATE_REDUCTION_TIME
              current_rate = current_rate * RATE_REDUCTION_COEFFICIENT / RATE_DENOMINATOR  # double-division with rounding made rate a bit less => good
              assert current_rate <= INITIAL_RATE  # This should never happen
      
          return to_mint
      
      
      @external
      def set_minter(_minter: address):
          """
          @notice Set the minter address
          @dev Only callable once, when minter has not yet been set
          @param _minter Address of the minter
          """
          assert msg.sender == self.admin  # dev: admin only
          assert self.minter == ZERO_ADDRESS  # dev: can set the minter only once, at creation
          self.minter = _minter
          log SetMinter(_minter)
      
      
      @external
      def set_admin(_admin: address):
          """
          @notice Set the new admin.
          @dev After all is set up, admin only can change the token name
          @param _admin New admin address
          """
          assert msg.sender == self.admin  # dev: admin only
          self.admin = _admin
          log SetAdmin(_admin)
      
      
      @external
      @view
      def totalSupply() -> uint256:
          """
          @notice Total number of tokens in existence.
          """
          return self.total_supply
      
      
      @external
      @view
      def allowance(_owner : address, _spender : address) -> uint256:
          """
          @notice Check the amount of tokens that an owner allowed to a spender
          @param _owner The address which owns the funds
          @param _spender The address which will spend the funds
          @return uint256 specifying the amount of tokens still available for the spender
          """
          return self.allowances[_owner][_spender]
      
      
      @external
      def transfer(_to : address, _value : uint256) -> bool:
          """
          @notice Transfer `_value` tokens from `msg.sender` to `_to`
          @dev Vyper does not allow underflows, so the subtraction in
               this function will revert on an insufficient balance
          @param _to The address to transfer to
          @param _value The amount to be transferred
          @return bool success
          """
          assert _to != ZERO_ADDRESS  # dev: transfers to 0x0 are not allowed
          self.balanceOf[msg.sender] -= _value
          self.balanceOf[_to] += _value
          log Transfer(msg.sender, _to, _value)
          return True
      
      
      @external
      def transferFrom(_from : address, _to : address, _value : uint256) -> bool:
          """
           @notice Transfer `_value` tokens from `_from` to `_to`
           @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
           @return bool success
          """
          assert _to != ZERO_ADDRESS  # dev: transfers to 0x0 are not allowed
          # NOTE: vyper does not allow underflows
          #       so the following subtraction would revert on insufficient balance
          self.balanceOf[_from] -= _value
          self.balanceOf[_to] += _value
          self.allowances[_from][msg.sender] -= _value
          log Transfer(_from, _to, _value)
          return True
      
      
      @external
      def approve(_spender : address, _value : uint256) -> bool:
          """
          @notice Approve `_spender` to transfer `_value` tokens on behalf of `msg.sender`
          @dev Approval may only be from zero -> nonzero or from nonzero -> zero in order
              to mitigate the potential race condition described here:
              https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
          @param _spender The address which will spend the funds
          @param _value The amount of tokens to be spent
          @return bool success
          """
          assert _value == 0 or self.allowances[msg.sender][_spender] == 0
          self.allowances[msg.sender][_spender] = _value
          log Approval(msg.sender, _spender, _value)
          return True
      
      
      @external
      def mint(_to: address, _value: uint256) -> bool:
          """
          @notice Mint `_value` tokens and assign them to `_to`
          @dev Emits a Transfer event originating from 0x00
          @param _to The account that will receive the created tokens
          @param _value The amount that will be created
          @return bool success
          """
          assert msg.sender == self.minter  # dev: minter only
          assert _to != ZERO_ADDRESS  # dev: zero address
      
          if block.timestamp >= self.start_epoch_time + RATE_REDUCTION_TIME:
              self._update_mining_parameters()
      
          _total_supply: uint256 = self.total_supply + _value
          assert _total_supply <= self._available_supply()  # dev: exceeds allowable mint amount
          self.total_supply = _total_supply
      
          self.balanceOf[_to] += _value
          log Transfer(ZERO_ADDRESS, _to, _value)
      
          return True
      
      
      @external
      def burn(_value: uint256) -> bool:
          """
          @notice Burn `_value` tokens belonging to `msg.sender`
          @dev Emits a Transfer event with a destination of 0x00
          @param _value The amount that will be burned
          @return bool success
          """
          self.balanceOf[msg.sender] -= _value
          self.total_supply -= _value
      
          log Transfer(msg.sender, ZERO_ADDRESS, _value)
          return True
      
      
      @external
      def set_name(_name: String[64], _symbol: String[32]):
          """
          @notice Change the token name and symbol to `_name` and `_symbol`
          @dev Only callable by the admin account
          @param _name New token name
          @param _symbol New token symbol
          """
          assert msg.sender == self.admin, "Only admin is allowed to change name"
          self.name = _name
          self.symbol = _symbol

      File 3 of 3: Vyper_contract
      # @version 0.2.15
      """
      @title StableSwap
      @author Curve.Fi
      @license Copyright (c) Curve.Fi, 2020-2021 - all rights reserved
      @notice 2 coin pool implementation with no lending
      @dev Optimized to only support ERC20's with 18 decimals that return True/revert
      """
      
      from vyper.interfaces import ERC20
      
      interface Factory:
          def convert_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 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
      
      
      N_COINS: constant(int128) = 2
      PRECISION: constant(int128) = 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)
      
      name: public(String[64])
      symbol: public(String[32])
      
      balanceOf: public(HashMap[address, uint256])
      allowance: public(HashMap[address, HashMap[address, uint256]])
      totalSupply: public(uint256)
      
      
      @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],
          _coins: address[4],
          _rate_multipliers: uint256[4],
          _A: uint256,
          _fee: uint256,
      ):
          """
          @notice Contract constructor
          @param _name Name of the new pool
          @param _symbol Token symbol
          @param _coins List of all ERC20 conract addresses of coins
          @param _rate_multipliers List of number of decimals in coins
          @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
      
          for i in range(N_COINS):
              coin: address = _coins[i]
              if coin == ZERO_ADDRESS:
                  break
              self.coins[i] = coin
              assert _rate_multipliers[i] == PRECISION
      
          A: uint256 = _A * A_PRECISION
          self.initial_A = A
          self.future_A = A
          self.fee = _fee
          self.factory = msg.sender
      
          self.name = concat("Curve.fi Factory Plain Pool: ", _name)
          self.symbol = concat(_symbol, "-f")
      
          # 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
          self.balanceOf[_from] -= _value
          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
      @external
      def get_balances() -> uint256[N_COINS]:
          return self.balances
      
      
      @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 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
          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 * D / _xp[0] * D / _xp[1] / (N_COINS)**2
              Dprev: uint256 = 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
      @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()
          D: uint256 = self.get_D(self.balances, 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()
          balances: uint256[N_COINS] = self.balances
      
          D0: uint256 = self.get_D(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(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()
          old_balances: uint256[N_COINS] = self.balances
      
          # Initial invariant
          D0: uint256 = self.get_D(old_balances, amp)
      
          total_supply: uint256 = self.totalSupply
          new_balances: uint256[N_COINS] = old_balances
          for i in range(N_COINS):
              amount: uint256 = _amounts[i]
              if total_supply == 0:
                  assert amount > 0  # dev: initial deposit requires all coins
              new_balances[i] += amount
      
          # Invariant after change
          D1: uint256 = self.get_D(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(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, "Slippage screwed you"
      
          # Take coins from the sender
          for i in range(N_COINS):
              amount: uint256 = _amounts[i]
              if amount > 0:
                  assert ERC20(self.coins[i]).transferFrom(msg.sender, self, amount)
      
          # 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:
          """
          Calculate x[j] if one makes x[i] = x
      
          Done by solving quadratic equation iteratively.
          x_1**2 + x_1 * (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 != 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
          """
          xp: uint256[N_COINS] = self.balances
      
          x: uint256 = xp[i] + dx
          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
      
      
      @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
          @return Actual amount of `j` received
          """
          old_balances: uint256[N_COINS] = self.balances
      
          x: uint256 = old_balances[i] + _dx
          y: uint256 = self.get_y(i, j, x, old_balances)
      
          dy: uint256 = old_balances[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_fee
          assert dy >= _min_dy, "Exchange resulted in fewer coins than expected"
      
          dy_admin_fee: uint256 = dy_fee * ADMIN_FEE / FEE_DENOMINATOR
      
          # 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
      
          assert ERC20(self.coins[i]).transferFrom(msg.sender, self, _dx)
          assert ERC20(self.coins[j]).transfer(_receiver, dy)
      
          log TokenExchange(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], "Withdrawal resulted in fewer coins than expected"
              self.balances[i] = old_balance - value
              amounts[i] = value
              assert ERC20(self.coins[i]).transfer(_receiver, value)
      
          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()
          old_balances: uint256[N_COINS] = self.balances
          D0: uint256 = self.get_D(old_balances, amp)
      
          new_balances: uint256[N_COINS] = old_balances
          for i in range(N_COINS):
              new_balances[i] -= _amounts[i]
          D1: uint256 = self.get_D(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(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, "Slippage screwed you"
      
          total_supply -= burn_amount
          self.totalSupply = total_supply
          self.balanceOf[msg.sender] -= burn_amount
          log Transfer(msg.sender, ZERO_ADDRESS, burn_amount)
      
          for i in range(N_COINS):
              if _amounts[i] != 0:
                  assert ERC20(self.coins[i]).transfer(_receiver, _amounts[i])
      
          log RemoveLiquidityImbalance(msg.sender, _amounts, fees, D1, total_supply)
      
          return burn_amount
      
      
      @pure
      @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 + x_1 * (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()
          balances: uint256[N_COINS] = self.balances
          D0: uint256 = self.get_D(balances, amp)
      
          total_supply: uint256 = self.totalSupply
          D1: uint256 = D0 - _burn_amount * D0 / total_supply
          new_y: uint256 = self.get_y_D(amp, i, balances, 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 = balances[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 = (balances[i] - new_y)  # w/o fees
          dy = (dy - 1)  # Withdraw less to account for rounding errors
      
          return [dy, dy_0 - dy]
      
      
      @view
      @external
      def calc_withdraw_one_coin(_burn_amount: uint256, i: int128, _previous: bool = False) -> 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, "Not enough coins removed"
      
          self.balances[i] -= (dy[0] + dy[1] * ADMIN_FEE / FEE_DENOMINATOR)
          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)
      
          assert ERC20(self.coins[i]).transfer(_receiver, dy[0])
      
          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:
          return ERC20(self.coins[i]).balanceOf(self) - self.balances[i]
      
      
      @external
      def withdraw_admin_fees():
          receiver: address = Factory(self.factory).get_fee_receiver(self)
      
          for i in range(N_COINS):
              coin: address = self.coins[i]
              fees: uint256 = ERC20(coin).balanceOf(self) - self.balances[i]
              ERC20(coin).transfer(receiver, fees)