2020-10-23 01:16:52 +08:00
|
|
|
|
/*
|
|
|
|
|
|
|
|
|
|
|
|
Copyright 2020 DODO ZOO.
|
|
|
|
|
|
SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
pragma solidity 0.6.9;
|
|
|
|
|
|
pragma experimental ABIEncoderV2;
|
|
|
|
|
|
|
2020-11-18 17:51:50 +08:00
|
|
|
|
import {DVMVault} from "./DVMVault.sol";
|
2020-10-23 01:16:52 +08:00
|
|
|
|
import {SafeMath} from "../../lib/SafeMath.sol";
|
|
|
|
|
|
import {DecimalMath} from "../../lib/DecimalMath.sol";
|
|
|
|
|
|
import {DODOMath} from "../../lib/DODOMath.sol";
|
2020-11-06 16:03:18 +08:00
|
|
|
|
import {IDODOCallee} from "../../intf/IDODOCallee.sol";
|
2020-11-18 17:51:50 +08:00
|
|
|
|
import {RState, PMMState, PMMPricing} from "../../lib/PMMPricing.sol";
|
2020-10-23 17:11:50 +08:00
|
|
|
|
|
2020-11-18 17:51:50 +08:00
|
|
|
|
contract DVMTrader is DVMVault {
|
2020-10-23 17:11:50 +08:00
|
|
|
|
using SafeMath for uint256;
|
2020-10-23 01:16:52 +08:00
|
|
|
|
|
2020-11-18 17:51:50 +08:00
|
|
|
|
// ============ Modifiers ============
|
|
|
|
|
|
|
|
|
|
|
|
modifier isBuyAllow(address trader) {
|
|
|
|
|
|
require(!_BUYING_CLOSE_ && _TRADE_PERMISSION_.isAllowed(trader), "TRADER_BUY_NOT_ALLOWED");
|
|
|
|
|
|
_;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
modifier isSellAllow(address trader) {
|
|
|
|
|
|
require(
|
|
|
|
|
|
!_SELLING_CLOSE_ && _TRADE_PERMISSION_.isAllowed(trader),
|
|
|
|
|
|
"TRADER_SELL_NOT_ALLOWED"
|
|
|
|
|
|
);
|
|
|
|
|
|
_;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
modifier limitGasPrice() {
|
|
|
|
|
|
require(tx.gasprice <= _GAS_PRICE_LIMIT_.get(), "GAS_PRICE_EXCEED");
|
|
|
|
|
|
_;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ============ Execute ============
|
|
|
|
|
|
|
2020-11-06 16:03:18 +08:00
|
|
|
|
function sellBase(address to)
|
|
|
|
|
|
external
|
|
|
|
|
|
preventReentrant
|
2020-11-11 16:42:00 +08:00
|
|
|
|
limitGasPrice
|
2020-11-06 16:03:18 +08:00
|
|
|
|
isSellAllow(to)
|
|
|
|
|
|
returns (uint256 receiveQuoteAmount)
|
|
|
|
|
|
{
|
2020-11-18 17:51:50 +08:00
|
|
|
|
uint256 baseInput = getBaseInput();
|
2020-10-23 17:11:50 +08:00
|
|
|
|
uint256 mtFee;
|
2020-11-18 17:51:50 +08:00
|
|
|
|
(receiveQuoteAmount, mtFee) = querySellBase(tx.origin, baseInput);
|
|
|
|
|
|
_transferQuoteOut(to, receiveQuoteAmount);
|
|
|
|
|
|
_transferQuoteOut(_MAINTAINER_, mtFee);
|
|
|
|
|
|
_sync();
|
2020-10-23 17:11:50 +08:00
|
|
|
|
return receiveQuoteAmount;
|
2020-10-23 01:16:52 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2020-11-06 16:03:18 +08:00
|
|
|
|
function sellQuote(address to)
|
|
|
|
|
|
external
|
|
|
|
|
|
preventReentrant
|
2020-11-11 16:42:00 +08:00
|
|
|
|
limitGasPrice
|
2020-11-06 16:03:18 +08:00
|
|
|
|
isBuyAllow(to)
|
|
|
|
|
|
returns (uint256 receiveBaseAmount)
|
|
|
|
|
|
{
|
2020-11-18 17:51:50 +08:00
|
|
|
|
uint256 quoteInput = getQuoteInput();
|
2020-10-23 17:11:50 +08:00
|
|
|
|
uint256 mtFee;
|
2020-11-18 17:51:50 +08:00
|
|
|
|
(receiveBaseAmount, mtFee) = querySellQuote(tx.origin, quoteInput);
|
|
|
|
|
|
_transferBaseOut(to, receiveBaseAmount);
|
|
|
|
|
|
_transferBaseOut(_MAINTAINER_, mtFee);
|
|
|
|
|
|
_sync();
|
2020-10-23 17:11:50 +08:00
|
|
|
|
return receiveBaseAmount;
|
2020-10-23 01:16:52 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2020-11-18 17:51:50 +08:00
|
|
|
|
// 这是一个试验性质的函数
|
|
|
|
|
|
// 没有走标准库,需要仔细考虑下
|
2020-11-05 00:26:45 +08:00
|
|
|
|
function flashLoan(
|
|
|
|
|
|
uint256 baseAmount,
|
|
|
|
|
|
uint256 quoteAmount,
|
|
|
|
|
|
address assetTo,
|
|
|
|
|
|
bytes calldata data
|
2020-11-06 16:03:18 +08:00
|
|
|
|
) external preventReentrant {
|
2020-11-18 17:51:50 +08:00
|
|
|
|
_transferBaseOut(assetTo, baseAmount);
|
|
|
|
|
|
_transferQuoteOut(assetTo, quoteAmount);
|
2020-11-06 16:03:18 +08:00
|
|
|
|
|
|
|
|
|
|
if (data.length > 0)
|
|
|
|
|
|
IDODOCallee(assetTo).DVMFlashLoanCall(msg.sender, baseAmount, quoteAmount, data);
|
2020-11-05 00:26:45 +08:00
|
|
|
|
|
2020-11-18 17:51:50 +08:00
|
|
|
|
(uint256 baseReserve, uint256 quoteReserve) = getVaultReserve();
|
|
|
|
|
|
(uint256 baseBalance, uint256 quoteBalance) = getVaultBalance();
|
2020-11-06 00:31:30 +08:00
|
|
|
|
|
2020-11-20 15:33:51 +08:00
|
|
|
|
uint256 mtFeeRate = _MT_FEE_RATE_MODEL_.getFeeRate(tx.origin);
|
|
|
|
|
|
uint256 lpFeeRate = _LP_FEE_RATE_MODEL_.getFeeRate(tx.origin);
|
2020-11-06 00:31:30 +08:00
|
|
|
|
if (baseBalance < baseReserve) {
|
|
|
|
|
|
uint256 validBaseOut = DecimalMath.divCeil(
|
|
|
|
|
|
baseReserve - baseBalance,
|
|
|
|
|
|
DecimalMath.ONE.sub(mtFeeRate).sub(lpFeeRate)
|
|
|
|
|
|
);
|
|
|
|
|
|
baseBalance = baseReserve.sub(validBaseOut);
|
2020-11-18 17:51:50 +08:00
|
|
|
|
_transferBaseOut(_MAINTAINER_, DecimalMath.mulCeil(validBaseOut, mtFeeRate));
|
2020-11-06 00:31:30 +08:00
|
|
|
|
}
|
|
|
|
|
|
if (quoteBalance < quoteReserve) {
|
|
|
|
|
|
uint256 validQuoteOut = DecimalMath.divCeil(
|
|
|
|
|
|
quoteReserve - quoteBalance,
|
|
|
|
|
|
DecimalMath.ONE.sub(mtFeeRate).sub(lpFeeRate)
|
|
|
|
|
|
);
|
|
|
|
|
|
quoteBalance = quoteReserve.sub(validQuoteOut);
|
2020-11-18 17:51:50 +08:00
|
|
|
|
_transferQuoteOut(_MAINTAINER_, DecimalMath.mulCeil(validQuoteOut, mtFeeRate));
|
2020-11-06 00:31:30 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
require(
|
|
|
|
|
|
calculateBase0(baseBalance, quoteBalance) >= calculateBase0(baseReserve, quoteReserve),
|
|
|
|
|
|
"FLASH_LOAN_FAILED"
|
|
|
|
|
|
);
|
|
|
|
|
|
|
2020-11-18 17:51:50 +08:00
|
|
|
|
_sync();
|
2020-11-05 00:26:45 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function querySellBase(address trader, uint256 payBaseAmount)
|
2020-10-23 17:11:50 +08:00
|
|
|
|
public
|
|
|
|
|
|
view
|
|
|
|
|
|
returns (uint256 receiveQuoteAmount, uint256 mtFee)
|
|
|
|
|
|
{
|
2020-11-18 17:51:50 +08:00
|
|
|
|
(receiveQuoteAmount, ) = PMMPricing.sellBaseToken(getPMMState(), payBaseAmount);
|
2020-11-05 00:26:45 +08:00
|
|
|
|
|
2020-11-06 00:31:30 +08:00
|
|
|
|
uint256 lpFeeRate = _LP_FEE_RATE_MODEL_.getFeeRate(trader);
|
|
|
|
|
|
uint256 mtFeeRate = _MT_FEE_RATE_MODEL_.getFeeRate(trader);
|
2020-11-18 17:51:50 +08:00
|
|
|
|
mtFee = DecimalMath.mulCeil(receiveQuoteAmount, mtFeeRate);
|
|
|
|
|
|
receiveQuoteAmount = DecimalMath.mulFloor(
|
|
|
|
|
|
receiveQuoteAmount,
|
|
|
|
|
|
DecimalMath.ONE.sub(mtFeeRate).sub(lpFeeRate)
|
|
|
|
|
|
);
|
2020-11-05 00:26:45 +08:00
|
|
|
|
|
2020-10-23 17:11:50 +08:00
|
|
|
|
return (receiveQuoteAmount, mtFee);
|
|
|
|
|
|
}
|
2020-10-23 01:16:52 +08:00
|
|
|
|
|
2020-11-05 00:26:45 +08:00
|
|
|
|
function querySellQuote(address trader, uint256 payQuoteAmount)
|
2020-10-23 17:11:50 +08:00
|
|
|
|
public
|
|
|
|
|
|
view
|
|
|
|
|
|
returns (uint256 receiveBaseAmount, uint256 mtFee)
|
|
|
|
|
|
{
|
2020-11-18 17:51:50 +08:00
|
|
|
|
(receiveBaseAmount, ) = PMMPricing.sellQuoteToken(getPMMState(), payQuoteAmount);
|
2020-11-05 00:26:45 +08:00
|
|
|
|
|
2020-11-06 00:31:30 +08:00
|
|
|
|
uint256 lpFeeRate = _LP_FEE_RATE_MODEL_.getFeeRate(trader);
|
|
|
|
|
|
uint256 mtFeeRate = _MT_FEE_RATE_MODEL_.getFeeRate(trader);
|
2020-11-18 17:51:50 +08:00
|
|
|
|
mtFee = DecimalMath.mulCeil(receiveBaseAmount, mtFeeRate);
|
|
|
|
|
|
receiveBaseAmount = DecimalMath.mulFloor(
|
2020-11-06 16:03:18 +08:00
|
|
|
|
receiveBaseAmount,
|
|
|
|
|
|
DecimalMath.ONE.sub(mtFeeRate).sub(lpFeeRate)
|
|
|
|
|
|
);
|
2020-11-18 17:51:50 +08:00
|
|
|
|
return (receiveBaseAmount, mtFee);
|
2020-11-06 16:03:18 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2020-11-18 17:51:50 +08:00
|
|
|
|
// // 这是一个仅供查询的合约,所有交易都是基于先给input,再输出output的
|
|
|
|
|
|
// // 所以想要买10ETH,这个函数可以给你一个大概的成本,你用这个成本输入,最后能否得到10ETH是要看情况的
|
|
|
|
|
|
// function queryBuyBase(address trader, uint256 receiveBaseAmount)
|
|
|
|
|
|
// public
|
|
|
|
|
|
// view
|
|
|
|
|
|
// returns (uint256 payQuoteAmount)
|
|
|
|
|
|
// {
|
|
|
|
|
|
// uint256 mtFeeRate = _MT_FEE_RATE_MODEL_.getFeeRate(trader);
|
|
|
|
|
|
// uint256 lpFeeRate = _LP_FEE_RATE_MODEL_.getFeeRate(trader);
|
|
|
|
|
|
// uint256 validReceiveBaseAmount = DecimalMath.divCeil(
|
|
|
|
|
|
// receiveBaseAmount,
|
|
|
|
|
|
// DecimalMath.ONE.sub(mtFeeRate).sub(lpFeeRate)
|
|
|
|
|
|
// );
|
|
|
|
|
|
// (uint256 baseReserve, uint256 quoteReserve) = getVaultReserve();
|
|
|
|
|
|
// require(baseReserve > validReceiveBaseAmount, "DODO_BASE_BALANCE_NOT_ENOUGH");
|
|
|
|
|
|
|
|
|
|
|
|
// uint256 B0 = calculateBase0(baseReserve, quoteReserve);
|
|
|
|
|
|
// uint256 B2 = baseReserve.sub(validReceiveBaseAmount);
|
|
|
|
|
|
// payQuoteAmount = DODOMath._GeneralIntegrate(B0, baseReserve, B2, _I_, _K_);
|
|
|
|
|
|
// return payQuoteAmount;
|
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
|
|
|
// function queryBuyQuote(address trader, uint256 receiveQuoteAmount)
|
|
|
|
|
|
// public
|
|
|
|
|
|
// view
|
|
|
|
|
|
// returns (uint256 payBaseAmount)
|
|
|
|
|
|
// {
|
|
|
|
|
|
// uint256 mtFeeRate = _MT_FEE_RATE_MODEL_.getFeeRate(trader);
|
|
|
|
|
|
// uint256 lpFeeRate = _LP_FEE_RATE_MODEL_.getFeeRate(trader);
|
|
|
|
|
|
// uint256 validReceiveQuoteAmount = DecimalMath.divCeil(
|
|
|
|
|
|
// receiveQuoteAmount,
|
|
|
|
|
|
// DecimalMath.ONE.sub(mtFeeRate).sub(lpFeeRate)
|
|
|
|
|
|
// );
|
|
|
|
|
|
// (uint256 baseReserve, uint256 quoteReserve) = getVaultReserve();
|
|
|
|
|
|
// require(quoteReserve > validReceiveQuoteAmount, "DODO_QUOTE_BALANCE_NOT_ENOUGH");
|
|
|
|
|
|
|
|
|
|
|
|
// uint256 B0 = calculateBase0(baseReserve, quoteReserve);
|
|
|
|
|
|
// uint256 fairAmount = DecimalMath.divFloor(validReceiveQuoteAmount, _I_);
|
|
|
|
|
|
// payBaseAmount = DODOMath._SolveQuadraticFunctionForTrade(
|
|
|
|
|
|
// B0,
|
|
|
|
|
|
// baseReserve,
|
|
|
|
|
|
// fairAmount,
|
|
|
|
|
|
// true,
|
|
|
|
|
|
// _K_
|
|
|
|
|
|
// );
|
|
|
|
|
|
// return payBaseAmount;
|
|
|
|
|
|
// }
|
2020-11-06 01:20:07 +08:00
|
|
|
|
|
2020-10-27 02:53:23 +08:00
|
|
|
|
function getMidPrice() public view returns (uint256 midPrice) {
|
2020-11-18 17:51:50 +08:00
|
|
|
|
(uint256 baseReserve, uint256 quoteReserve) = getVaultReserve();
|
2020-11-05 14:10:59 +08:00
|
|
|
|
uint256 B0 = calculateBase0(baseReserve, quoteReserve);
|
2020-11-05 00:26:45 +08:00
|
|
|
|
|
|
|
|
|
|
uint256 offsetRatio = DecimalMath.ONE.mul(B0).div(baseReserve).mul(B0).div(baseReserve);
|
2020-10-27 02:53:23 +08:00
|
|
|
|
uint256 offset = DecimalMath.ONE.sub(_K_).add(DecimalMath.mulFloor(offsetRatio, _K_));
|
|
|
|
|
|
return DecimalMath.mulFloor(_I_, offset);
|
|
|
|
|
|
}
|
2020-11-18 17:51:50 +08:00
|
|
|
|
|
|
|
|
|
|
// ============ Helper Functions ============
|
|
|
|
|
|
|
|
|
|
|
|
function getPMMState() public view returns (PMMState memory state) {
|
|
|
|
|
|
state.i = _I_;
|
|
|
|
|
|
state.K = _K_;
|
|
|
|
|
|
state.B = _BASE_RESERVE_;
|
|
|
|
|
|
state.Q = _QUOTE_RESERVE_;
|
|
|
|
|
|
state.B0 = calculateBase0(state.B, state.Q);
|
|
|
|
|
|
state.Q0 = 0;
|
|
|
|
|
|
state.R = RState.ABOVE_ONE;
|
|
|
|
|
|
return state;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function calculateBase0(uint256 baseAmount, uint256 quoteAmount) public view returns (uint256) {
|
|
|
|
|
|
uint256 fairAmount = DecimalMath.divFloor(quoteAmount, _I_);
|
|
|
|
|
|
return DODOMath._SolveQuadraticFunctionForTarget(baseAmount, _K_, fairAmount);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function getBase0() public view returns (uint256) {
|
|
|
|
|
|
(uint256 baseAmount, uint256 quoteAmount) = getVaultReserve();
|
|
|
|
|
|
uint256 fairAmount = DecimalMath.divFloor(quoteAmount, _I_);
|
|
|
|
|
|
return DODOMath._SolveQuadraticFunctionForTarget(baseAmount, _K_, fairAmount);
|
|
|
|
|
|
}
|
2020-10-23 17:11:50 +08:00
|
|
|
|
}
|