Bitcoin logos on a pile of black boxes.
How to CreateFarm Contracts
December 6th, 2020

Overview of Liquidity Mining

Liquidity mining is one of the hottest topics in decentralized finance. Liquidity generally refers to the assets in an ecosystem.

In the case of governance protocols, liquidity also represents your piece of the governance pie, so to speak.

In this tutorial, I will review how I constructed a liquidity mine for a cryptocurrency project and how you may replicate.

We narrow it down to 3 simple steps: Import, Declare, and Design.


Step Zero (0): Planning Stage

The benefit of providing a liquidity mine is this foregoes the need for an ICO, where tokens are purchased directly from the project, which is subject to regulatory intervention. Ultimately, there should be a reason why you are utilizing this method to distribute tokens and you should consider the following before you jump into coding the smart contract itself. Some considerations you may have are as follows:

  • Distribution Method: consider whether you will be minting tokens or whether there will be another functionality you utilize to distribute tokens.
  • Inflation Rate: at what rate will your token be introduced into the ecosystem and for how long?
  • Utility: while this is not apparent in the contract itself, you should ask yourself: what is the purpose of the token you are creating?

In order to build a liquidity mine you need to first determine whether your mine will be minting or transferring tokens as a reward to miners.

In the case of the DAO, we wanted to ensure we eliminate the possibility of minting additional tokens, thus I had to come up with a way to transfer tokens from the contract itself to the users of the dApp.

Ordinarily, as is the case with Sushi, for example, one would mint the tokens as this is how the tokens are made available to the ecosystem.

Given the historical roots of the project I created this mine for (as the response to an unfortunate rug pull by CBDAO), we wanted to remove all doubt entirely, which is why we decidedly eliminated the mint function from the ERC20 contract.

Let’s get into the steps I followed to implement the smart contract that will lay the foundation for the Treasury Fund, launching December 22nd, 2020.


Step One (1): Import Libraries

  • IERC20.sol returns key aspects of the smart contract, for example, we use the transferFrom function to transfer (address sender, address recipient, uint256 amount).

  • SafeERC20.sol functions wrap amount ERC20 native functions that throw on failure (i.e. when the return is false). Your contract may inherent this functionality by simply including the line: “using SafeERC20 for IERC20”. EnumerableSet.sol enables you to store unique values, which is a handy tool in solidity, as you will see later in this article.

  • SafeMath.sol enables Solidity to throw an error in the case of overflow, eliminating an entire class of bugs.

  • Ownable.sol is a contract module which provides a basic access control mechanism where the owner may be granted access to a suite of functions inaccessible by others via the modifier ‘onlyOwner’.


Step Two (2): Declare Key Variables

You are welcome to declare however many variables are relevant for your project. In the case of the DAO, we are utilizing the Liquidity Mine as both a means to distribute the DAO governance tokens, but also as the launchpad for the the DAO Treasury fund.

GovTreasurer.sol
contract GovTreasurer is Ownable {
    using SafeMath for uint256;
    using SafeERC20 for IERC20;

    // INFO | USER VARIABLES
    struct UserInfo {
        uint256 amount;     // How many tokens the user has provided.
        uint256 rewardDebt; // Reward debt.
    }

    // INFO | POOL VARIABLES
    struct PoolInfo {
        IERC20 token;           // Address of token contract.
        uint256 allocPoint;       // How many allocation points assigned to this pool. GDAOs to distribute per block.
        uint256 taxRate;          // Rate at which the LP token is taxed.
        uint256 lastRewardBlock;  // Last block number that GDAOs distribution occurs.
        uint256 accGDAOPerShare; // Accumulated GDAOs per share, times 1e12. See below.
    }

As such, we include taxRate to represent the rate at which the pool is taxed. Note: due to the nature of Solidity, I had to get creative with this number, so it is not as simple as tax rate = 0.02, as Solidity does not cooperate with decimals. As such, the taxRate = 50 and we simply divide the value being deposited by 50 to acquire 2%. This is true for any integer n. Since every liquidity mining operation is unique, I left out a few lines as they consist of declaring a myriad of variables that may or may not be relevant to your project.


Step Three (3): Design Key Functions

Time to get to the functionality of the contract, which is stored in, you guessed it — your functions.

Let’s explore a few options for you to consider using the sample code snippet from the the DAO Liquidity Mining Smart Contract.

We start off by declaring a function named “deposit”, which accepts two input variables: Pool ID and Amount.

  • Pool ID the pool selected by the depositor.
  • Amount the amount of tokens to deposit.

Do not be fooled, this contract is unlike your typical deposit function as it is designed to acquire a tax of 2% on deposit, as discussed before.

  • taxedAmount: the amount that is taxed, such that taxedAmount = 2% of deposit amount.

This new variable is used 4 times in the function, as a deduction from the amount deposited to the smart contract, as the amount sent to the DAO treasury, as an update to the amount the user is owed back upon withdraw, and as a deduction from the amount that is emitted from the Deposit event.

Altogether, this function enables the desired effect. Get creative and design a function of your own.

Below is a snippet from the original deposit function:

deposit(pid,amount)
function deposit(uint256 _pid, uint256 _amount) public {
        PoolInfo storage pool = poolInfo[_pid];
        UserInfo storage user = userInfo[_pid][msg.sender];
        updatePool(_pid);
        uint256 taxedAmount = _amount.div(pool.taxRate);

        if (user.amount > 0) { // if there are already some amount deposited
            uint256 pending = user.amount.mul(pool.accGDAOPerShare).div(1e12).sub(user.rewardDebt);
            if(pending > 0) { // sends pending rewards, if applicable
                safeGDAOTransfer(msg.sender, pending);
            }
        }
        
        if(_amount > 0) { // if adding more
            pool.token.safeTransferFrom(address(msg.sender), address(this), _amount.sub(taxedAmount));
            pool.token.safeTransferFrom(address(msg.sender), address(treasury), taxedAmount);
            user.amount = user.amount.add(_amount.sub(taxedAmount)); // update user.amount = non-taxed amount
        }
        
        user.rewardDebt = user.amount.mul(pool.accGDAOPerShare).div(1e12);
        emit Deposit(msg.sender, _pid, _amount.sub(taxedAmount));
    }