Skip to content

HyunggyuJang/zap-in-service

Repository files navigation

Zap In Service

Welcome to the Zap In Service project! This innovative solution is designed to streamline the liquidity provision process on Uniswap V2. It enables liquidity providers to deposit a single token into a pool and receive Uniswap V2 LP tokens in return. This project addresses the common inconvenience faced by liquidity providers who typically need to deposit both tokens in a pool. It simplifies the process, making it more accessible for users, especially those new to DeFi, to participate in liquidity provision.

Getting Started

This section will guide you through the prerequisites, installation process, and how to run tests for the Zap In Service project.

Prerequisites

Before you begin, ensure you have the following installed:

  • Node.js (version 18)
  • npm (usually comes with Node.js)
  • Hardhat for compiling and running the smart contracts on a local blockchain
  • Poetry for managing Python dependencies (for Slither static analysis)

Installation

Follow these steps to set up the project locally:

  1. Clone the repository:
    git clone https://github.com/HyunggyuJang/zap-in-service.git
  2. Navigate to the project directory:
    cd zap-in-service
  3. Install the required npm packages:
    npm install
  4. Install the required Poetry packages:
     poetry install
  5. Set up your environment variables by creating a .env file in the root directory and adding the following:
    # .env
    INFURA_API_KEY=<your_infura_api_key>
    BLOCK_NUMBER=<block_number_you_want_to_fork>
    The INFURA_API_KEY is required to fork the Ethereum mainnet using Hardhat. You can obtain an API key by signing up on the Infura website. The BLOCK_NUMBER is the block number you want to fork from the Ethereum mainnet.

Run Tests

To ensure the smart contracts work as expected, run the provided test scripts with Hardhat:

npm run test

Run Slither Static Analysis

To perform static analysis on the smart contracts, run the following command:

npm run analyze

Run Local Blockchain & Deploy Contracts

You can run a local blockchain and deploy the smart contracts using Hardhat. To do so, run the following command:

npx hardhat node
npm run test -- --network localhost

This will start a local blockchain and deploy the smart contracts to the network by running the test scripts.

Design Notes

This section provides an overview of the smart contracts and the design decisions made for the Zap In Service project.

A naive approach to the Zap In Service would be to 1) split the input token amount into two equal parts, 2) swap one part for the other token using the target pool's liquidity, and 3) add the two tokens to the pool. However, this approach has several drawbacks, such as inevitable slippage and altering the pool's price, leading to a suboptimal solution where the user receives fewer LP tokens than expected.

A better approach is to use close-to-optimal solutions, such as Uniswap V3's Swap and Add functionality, which aggregates swap routes to minimize slippage and maximize the amount of liquidity added to the pool. However, this approach requires a lot of bootstrapping and is not feasible for the current Zap In Service project.

The current implementation of the Zap In Service is based on the following observations:

  • We can calculate the exact output amount of the output token by using Uniswap V2's formula.

    For example, the following code snippet is taken from the UniSwapV2Library.sol:

      function getAmountOut(
          uint256 amountIn,
          uint256 reserveIn,
          uint256 reserveOut
      ) internal pure returns (uint256 amountOut) {
          require(amountIn > 0, "UniswapV2Library: INSUFFICIENT_INPUT_AMOUNT");
          require(reserveIn > 0 && reserveOut > 0, "UniswapV2Library: INSUFFICIENT_LIQUIDITY");
          uint256 amountInWithFee = amountIn.mul(997);
          uint256 numerator = amountInWithFee.mul(reserveOut);
          uint256 denominator = reserveIn.mul(1000).add(amountInWithFee);
          amountOut = numerator / denominator;
      }

    The governing formula is the famous $x \times y = k$, where $x$ and $y$ are the reserve amounts of the input and output tokens, respectively, and $k$ is the product of the two reserve amounts. One twist is that Uniswap V2 applies a 0.3% fee to the input token amount, which is reflected in the amountInWithFee variable. The getAmountOut function calculates the exact output amount of the output token given the input amount and the reserve amounts of the input and output tokens, assuming underlying ERC20 tokens are not rebasing tokens.

  • Similarly, we can formulate the optimal ratio of the input token amount to the output token amount.

    Let's assume $a_{io}$ is the original input token amount, i.e., the amount of the single token the user wants to deposit into the pool, $a_i$ is the input token amount to be swapped for the output token, and $a_o$ is the output token amount to be received. Then the optimal ratio of $a_{io} - a_i$ to $a_o$ is given by the following formula: $$\frac{a_{io} - a_i}{a_o} = \frac{r_{in}}{r_{on}}$$ where $r_{in}$ and $r_{on}$ are the reserve amounts of the input and output tokens, after the input token amount $a_i$ is swapped for the output token.

    Note that $a_o$ can be expressed as a function of $a_i$ and $r_i$ and $r_o$, where $r_i$ and $r_o$ are the reserve amounts of the input and output tokens, respectively, before the input token amount $a_i$ is swapped for the output token, which are the known values. Also, $r_{in}$ and $r_{on}$ can be expressed as a function of $a_i$, $a_o$, $r_i$ and $r_o$ as follows: $$r_{in} = r_i + a_i,\quad r_{on} = r_o - a_o$$

    Therefore, we can solve the optimal ratio condition with respect to $a_i$, and it boils down to solving a quadratic equation: $$c a_i^2 - (1 + c) r_i a_i - a_{io} r_i = 0$$ where $c$ is the fee-adjusted constant, i.e., $c = 997/1000$. Since this form of quadratic equation has a unique positive solution, we can calculate $a_i$ solely based on $a_{io}$, $r_i$.

    In the code implementation, the relevant function is calculateSwapInAmount in the Helper.sol contract.

A space for improvement is to implement the aforementioned Uniswap V3's Swap and Add functionality and compare the performance based on the estimated LP token amount in return from off-chain computation, such as from client-side front-end, and choose the better one.

Time Spent on the Project

The project involved extensive research and implementation phases, including:

  • Approximately 5 hours researching relevant features available in open-source projects.
  • About 2 hours for theoretical research on Uniswap V2's formula and optimal token amount ratios.
  • Around 2 hours researching optimal implementation strategies.
  • Approximately 3 hours for the implementation of the smart contracts and tests.
  • About 2 hours for documentation, including this README.

The focus was on design and research to ensure an efficient and secure implementation, leveraging battle-tested solutions from the open-source community. The implementation phase was relatively straightforward, following the completion of the design and research phases.

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published