Documentation written by donchate
Note: this smart contract is only available on Hive Engine, there is no Steem Engine equivalent.
- Introduction
- Administrative Actions
- Swapping Actions
- Liquidity Actions
- Rewarding Liquidity Providers
- Interface Integration
- Tables available
This contract implements a constant product formula (CPF) automated market maker (AMM). The AMM holds quantities of Hive Engine tokens whose relative price is measured according to its reserves. A constant product pricing function maps the quantities of the assets in reserves to their marginal price. Hive Engine token holders may trade using this smart contract at the price specified by the pricing function, which is continually updated as reserves change after each trade. They may also provide liquidity to allow for a certain pair of tokens to be traded.
Simplified pricing function: k = x * y
where k is a constant
x, y are the available token reserves at time of trade
A price quote is defined as the price of one token in terms of another. Throughout the contract we show two tokens, the base token, which appears first and the quote token, which appears last. The price of the first token is always reflected in units of the second token.
A trading pair must be created before liquidity can be added and trades made.
Only one pair may exist for any two tokens, whether it is defined as TOKEN1:TOKEN2
or TOKEN2:TOKEN1
is set by the initial creator.
A fee of 1000 BEE is required.
-
requires active key: yes
-
can be called by: anyone
-
parameters:
- tokenPair (string): Trading pair name describing the two tokens that will be paired in the format
TOKEN1:TOKEN2
- tokenPair (string): Trading pair name describing the two tokens that will be paired in the format
-
example:
{
"tokenPair": "GLD:SLV"
}
- requires active key: yes
- can be called by: anyone
- parameters:
- tokenPair (string): Trading pair name describing the two tokens that will be paired in the format
TOKEN1:TOKEN2
- tokenSymbol (string): Valid and configured token symbol that is being traded per the
tradeType
- tokenOut (string): Amount of tokenSymbol that is being traded per the
tradeType
- tradeType (string): Either
exactInput
orexactOutput
indicating the type of trade - minAmountOut (string) (optional): Set minimum expected amount out for an
exactInput
trade (slippage protection) - maxAmountIn (string) (optional): Set maximum expected amount in for an
exactOutput
trade (slippage protection)
- tokenPair (string): Trading pair name describing the two tokens that will be paired in the format
A trader may wish to receive automatically traded tokens for an exact amount of input tokens. In the example below: 1 GLD is requested out of the swap and the amount required will be automatically calculated and deducted from the holder's token balance.
exactOutput
example:
{
"tokenPair": "GLD:SLV",
"tokenSymbol": "GLD",
"tokenAmount": "1",
"tradeType": "exactOutput",
"maxAmountIn": "1",
"isSignedWithActiveKey": true
}
A trader may wish to send an exact amount of tokens to the swap contract, and receive an automatically traded amount of tokens back. In the example below: 1 GLD is the amount being sent to the contract and a relative amount of SLV will be returned.
exactInput
example:
{
"tokenPair": "GLD:SLV",
"tokenSymbol": "GLD",
"tokenAmount": "1",
"tradeType": "exactInput",
"minAmountOut": "1",
"isSignedWithActiveKey": true
}
To provide the resources for trading to occur, liquidity must be sent and stored by the smart contract.
This action will send both tokens in a given pair to the smart contract to provide liquidity for traders to swap tokens.
If this is being called on a newly created market pool, the initial liquidity provider sets the starting price for the pair. If the pair tokens have orders in the HE order book, the initial price will be compared to the last price on the order book market for deviation beyond 1%. This is to avoid an unintentional arbitrage situation.
For existing pools, liquidity can only be added in such a way that the trading price is maintained. As such, the contract may adjust either base or quote quantity by a small factor to meet these requirements. If the adjustment exceeds the default or specified maxPriceImpact
, the liquidity will not be added.
If the provider is seeding a new pool, the number of shares they will receive will equal sqrt(x * y)
, where x and y represent the amount of each token provided.
-
requires active key: yes
-
can be called by: anyone
-
parameters:
- tokenPair (string): Trading pair name describing the two tokens that will be paired in the format
TOKEN1:TOKEN2
- baseQuantity (string): Amount to deposit into the base token reserve (first token in pair)
- quoteQuantity (string): Amount to deposit into the quote token reserve (second token in pair)
- maxPriceImpact (string) (optional): Amount of tolerance to price impact after adding liquidity. If unspecified, the default value is 1%. Max precision 3.
- maxDeviation (string) (optional): Amount of tolerance to price difference versus the regular HE market for the pair. Must be an integer equal to or greater than 0. To disable this check, set this value to '0'. If unspecified, the default value is 1%.
- tokenPair (string): Trading pair name describing the two tokens that will be paired in the format
-
example:
{
"tokenPair": "GLD:SLV",
"baseQuantity": "1000",
"quoteQuantity": "16000",
"maxPriceImpact": 1,
"maxDeviation": 1,
"isSignedWithActiveKey": true
}
This action allows a liquidity provider to withdraw their tokens from the market pool.
-
requires active key: yes
-
can be called by: anyone
-
parameters:
- tokenPair (string): Trading pair name describing the two tokens that will be paired in the format
TOKEN1:TOKEN2
- sharesOut (string): Percentage > 0 <= 100 - amount of liquidity shares to convert into tokens. Max precision 3
- tokenPair (string): Trading pair name describing the two tokens that will be paired in the format
-
example:
{
"tokenPair": "GLD:SLV",
"sharesOut": "50",
"isSignedWithActiveKey": true
}
Its possible to reward liquidity providers to the pool directly on the sidechain. This allows some customization for each pool and for more methods to be added in the future.
This action allows a token issuer to create a reward system fulfilled through the Hive Engine Mining Contract. The pool is automatically activated upon creation. A fee of 1000 BEE is required.
-
requires active key: yes
-
can be called by: token issuer
-
parameters:
- tokenPair (string): Trading pair name describing the two tokens that will be paired in the format
TOKEN1:TOKEN2
- lotteryWinners (integer >= 1 and <= 20): Number of lottery winners per round.
- lotteryIntervalHours (integer >= 1 and <= 720): How often in hours to run a lottery.
- lotteryAmount (string): Amount to pay out per round. Split among lotteryWinners winners.
- minedToken (string): Which token to issue as reward.
- tokenPair (string): Trading pair name describing the two tokens that will be paired in the format
-
example (only the following parameters are currently supported):
{
"tokenPair": "GLD:SLV",
"lotteryWinners": 20,
"lotteryIntervalHours": 1,
"lotteryAmount": "1",
"minedToken": "GLD",
"isSignedWithActiveKey": true
}
Disable or enable a mining pool. Lotteries will not occur if not active. No fee is required to change the status.
-
requires active key: yes
-
can be called by: token issuer
-
parameters:
- tokenPair (string): Trading pair name describing the two tokens that will be paired in the format
TOKEN1:TOKEN2
- minedToken (string): Which token to issue as reward.
- active (boolean): Set reward pool to active or inactive
- tokenPair (string): Trading pair name describing the two tokens that will be paired in the format
-
example:
{
"tokenPair": "GLD:SLV",
"minedToken": "GLD",
"active": true,
"isSignedWithActiveKey": true
}
To estimate the minimum token amount required to get an exact number of tokens out of a swap (exactOutput
):
const num = api.BigNumber(liquidityIn).times(amountOut);
const den = api.BigNumber(liquidityOut).minus(amountOut);
const amountIn = num.dividedBy(den);
To estimate the minimum token amount received when sending an exact number of tokens into a swap (exactInput
):
const num = api.BigNumber(amountIn).times(liquidityOut);
const den = api.BigNumber(liquidityIn).plus(amountIn);
const amountOut = num.dividedBy(den);
The trader's tolerage to slippage in percent should be determined to set the value for minAmountOut
or maxAmountIn
depending on trade direction.
To ensure that the addition of new liquidity doesn't move the price, a user can only add to both tokens in the pair in quantities that maintain the same price. Note: This does not apply to the initial liquidity provider, who sets the price if none is available from the internal oracle
const price = api.BigNumber(pool.quoteQuantity).dividedBy(pool.baseQuantity);
Liquidity is removed by specifying a % of shares to remove. The contract calculates the amount returned of each token based on the position shares relative to the pool totalShares
.
The amount of tokens returned can be estimated by the following formulas:
const baseOut = api.BigNumber(sharesDelta).times(pool.baseQuantity).dividedBy(pool.totalShares).toFixed(baseToken.precision, api.BigNumber.ROUND_DOWN);
const quoteOut = api.BigNumber(sharesDelta).times(pool.quoteQuantity).dividedBy(pool.totalShares).toFixed(quoteToken.precision, api.BigNumber.ROUND_DOWN);
contains internal contract parameters
- fields
- poolCreationFee = the cost in BEE to start a market pool for a token pair
contains the instance configurations
- fields
- *_id = MongoDB internal primary key
- *tokenPair = market pool token pair
- baseQuantity = base token reserve
- quoteQuantity = quote token reserve
- basePrice = trading price base per quote (usual way of reading a pair)
- quotePrice = trading price quote per base (reverse price)
- baseVolume = total base tokens traded
- quoteVolume = total quote tokens traded
- totalShares = total shares issued to liquidity providers
- precision = market pool token precision (maximum of either pair)
- creator = account name of initial pool creator
- fields
- _id = MongoDB internal primary key
- account = account name of the LP
- tokenPair = token pair in which there is a LP
- shares = number of shares of the liquidity pool