文件 1 的 1:RARESwapRouter.sol
pragma solidity =0.6.6;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes memory) {
this;
return msg.data;
}
}
pragma solidity =0.6.6;
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor() public{
_transferOwnership(_msgSender());
}
function owner() public view virtual returns (address) {
return _owner;
}
modifier onlyOwner() {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
_;
}
function renounceOwnership() public virtual onlyOwner {
_transferOwnership(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_transferOwnership(newOwner);
}
function _transferOwnership(address newOwner) internal virtual {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
pragma solidity >=0.6.2;
interface IRARESwapRouter01 {
function factory() external pure returns (address);
function WETH() external pure returns (address);
function addLiquidity(
address tokenA,
address tokenB,
uint amountADesired,
uint amountBDesired,
uint amountAMin,
uint amountBMin,
address to,
uint deadline
) external returns (uint amountA, uint amountB, uint liquidity);
function addLiquidityETH(
address token,
uint amountTokenDesired,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline
) external payable returns (uint amountToken, uint amountETH, uint liquidity);
function removeLiquidity(
address tokenA,
address tokenB,
uint liquidity,
uint amountAMin,
uint amountBMin,
address to,
uint deadline
) external returns (uint amountA, uint amountB);
function removeLiquidityETH(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline
) external returns (uint amountToken, uint amountETH);
function removeLiquidityWithPermit(
address tokenA,
address tokenB,
uint liquidity,
uint amountAMin,
uint amountBMin,
address to,
uint deadline,
bool approveMax, uint8 v, bytes32 r, bytes32 s
) external returns (uint amountA, uint amountB);
function removeLiquidityETHWithPermit(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline,
bool approveMax, uint8 v, bytes32 r, bytes32 s
) external returns (uint amountToken, uint amountETH);
function swapExactTokensForTokens(
uint amountIn,
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external returns (uint[] memory amounts);
function swapTokensForExactTokens(
uint amountOut,
uint amountInMax,
address[] calldata path,
address to,
uint deadline
) external returns (uint[] memory amounts);
function swapExactETHForTokens(uint amountOutMin, address[] calldata path, address to, uint deadline)
external
payable
returns (uint[] memory amounts);
function swapTokensForExactETH(uint amountOut, uint amountInMax, address[] calldata path, address to, uint deadline)
external
returns (uint[] memory amounts);
function swapExactTokensForETH(uint amountIn, uint amountOutMin, address[] calldata path, address to, uint deadline)
external
returns (uint[] memory amounts);
function swapETHForExactTokens(uint amountOut, address[] calldata path, address to, uint deadline)
external
payable
returns (uint[] memory amounts);
function quote(uint amountA, uint reserveA, uint reserveB) external pure returns (uint amountB);
function getAmountOut(uint amountIn, uint reserveIn, uint reserveOut) external pure returns (uint amountOut);
function getAmountIn(uint amountOut, uint reserveIn, uint reserveOut) external pure returns (uint amountIn);
function getAmountsOut(uint amountIn, address[] calldata path) external view returns (uint[] memory amounts);
function getAmountsIn(uint amountOut, address[] calldata path) external view returns (uint[] memory amounts);
}
pragma solidity >=0.6.2;
interface IRARESwapRouter is IRARESwapRouter01 {
function removeLiquidityETHSupportingFeeOnTransferTokens(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline
) external returns (uint amountETH);
function removeLiquidityETHWithPermitSupportingFeeOnTransferTokens(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline,
bool approveMax, uint8 v, bytes32 r, bytes32 s
) external returns (uint amountETH);
function swapExactTokensForTokensSupportingFeeOnTransferTokens(
uint amountIn,
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external;
function swapExactETHForTokensSupportingFeeOnTransferTokens(
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external payable;
function swapExactTokensForETHSupportingFeeOnTransferTokens(
uint amountIn,
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external;
}
pragma solidity >=0.5.0;
interface IToken {
function addPair(address pair, address token) external;
function depositLPFee(uint amount, address token) external;
}
pragma solidity >=0.5.0;
interface IWETH {
function deposit() external payable;
function transfer(address to, uint value) external returns (bool);
function withdraw(uint) external;
}
pragma solidity >=0.5.0;
interface IERC20 {
event Approval(address indexed owner, address indexed spender, uint value);
event Transfer(address indexed from, address indexed to, uint value);
function name() external view returns (string memory);
function symbol() external view returns (string memory);
function decimals() external view returns (uint8);
function totalSupply() external view returns (uint);
function balanceOf(address owner) external view returns (uint);
function allowance(address owner, address spender) external view returns (uint);
function approve(address spender, uint value) external returns (bool);
function transfer(address to, uint value) external returns (bool);
function transferFrom(address from, address to, uint value) external returns (bool);
}
pragma solidity =0.6.6;
library SafeMath {
function add(uint x, uint y) internal pure returns (uint z) {
require((z = x + y) >= x, 'ds-math-add-overflow');
}
function sub(uint x, uint y) internal pure returns (uint z) {
require((z = x - y) <= x, 'ds-math-sub-underflow');
}
function mul(uint x, uint y) internal pure returns (uint z) {
require(y == 0 || (z = x * y) / y == x, 'ds-math-mul-overflow');
}
function div(uint x, uint y) internal pure returns (uint z) {
require(y > 0, "ds-math-div-underflow");
z = x / y;
}
}
pragma solidity >=0.5.0;
interface IRARESwapPair {
event Approval(address indexed owner, address indexed spender, uint value);
event Transfer(address indexed from, address indexed to, uint value);
function baseToken() external view returns (address);
function getTotalFee() external view returns (uint);
function name() external pure returns (string memory);
function symbol() external pure returns (string memory);
function decimals() external pure returns (uint8);
function totalSupply() external view returns (uint);
function balanceOf(address owner) external view returns (uint);
function allowance(address owner, address spender) external view returns (uint);
function updateTotalFee(uint totalFee) external returns (bool);
function approve(address spender, uint value) external returns (bool);
function transfer(address to, uint value) external returns (bool);
function transferFrom(address from, address to, uint value) external returns (bool);
function DOMAIN_SEPARATOR() external view returns (bytes32);
function PERMIT_TYPEHASH() external pure returns (bytes32);
function nonces(address owner) external view returns (uint);
function permit(address owner, address spender, uint value, uint deadline, uint8 v, bytes32 r, bytes32 s) external;
event Mint(address indexed sender, uint amount0, uint amount1);
event Burn(address indexed sender, uint amount0, uint amount1, address indexed to);
event Swap(
address indexed sender,
uint amount0In,
uint amount1In,
uint amount0Out,
uint amount1Out,
address indexed to
);
event Sync(uint112 reserve0, uint112 reserve1);
function MINIMUM_LIQUIDITY() external pure returns (uint);
function factory() external view returns (address);
function token0() external view returns (address);
function token1() external view returns (address);
function getReserves() external view returns (uint112 reserve0, uint112 reserve1, uint32 blockTimestampLast, address _baseToken);
function price0CumulativeLast() external view returns (uint);
function price1CumulativeLast() external view returns (uint);
function kLast() external view returns (uint);
function mint(address to) external returns (uint liquidity);
function burn(address to) external returns (uint amount0, uint amount1);
function swap(uint amount0Out, uint amount1Out, uint amount0Fee, uint amount1Fee, address to, bytes calldata data) external;
function skim(address to) external;
function sync() external;
function initialize(address, address) external;
function setBaseToken(address _baseToken) external;
}
pragma solidity >=0.5.0;
interface IRARESwapFactory {
event PairCreated(address indexed token0, address indexed token1, address pair, uint);
function feeTo() external view returns (address);
function feeToSetter() external view returns (address);
function getPair(address tokenA, address tokenB) external view returns (address pair);
function allPairs(uint) external view returns (address pair);
function allPairsLength() external view returns (uint);
function pairExist(address pair) external view returns (bool);
function createPair(address tokenA, address tokenB) external returns (address pair);
function setFeeTo(address) external;
function setFeeToSetter(address) external;
function routerInitialize(address) external;
function routerAddress() external view returns (address);
}
pragma solidity =0.6.6;
abstract contract FeeStore is Ownable {
uint public adminFee;
address public adminFeeAddress;
address public factoryAddress;
mapping (address => address) public pairFeeAddress;
function initialize(address _factory, uint256 _adminFee, address _adminFeeAddress) internal {
factoryAddress = _factory;
adminFee = _adminFee;
adminFeeAddress = _adminFeeAddress;
}
function feeAdddressSetWhileSwap(address pair,address tokenAddress) public onlyOwner {
require(IRARESwapFactory(factoryAddress).pairExist(pair), "RARESwap FeeStore: Pair is not Exist");
require(IRARESwapPair(pair).token0() == tokenAddress || IRARESwapPair(pair).token1() == tokenAddress, "RARESwap FeeStore: Invalid token address");
pairFeeAddress[pair] = tokenAddress;
}
function feeAddressGet() public view returns (address) {
return (adminFeeAddress == address(0) ? address(this) : adminFeeAddress);
}
function setAdminFee (address _adminFeeAddress, uint _adminFee) external onlyOwner {
require (_adminFee <= 100, "RARESwap: Fee exceeds 1%");
adminFeeAddress = _adminFeeAddress;
adminFee = _adminFee;
}
}
pragma solidity >=0.5.0;
library RARESwapLibrary {
using SafeMath for uint;
function sortTokens(address tokenA, address tokenB) internal pure returns (address token0, address token1) {
require(tokenA != tokenB, 'RARESwapLibrary: IDENTICAL_ADDRESSES');
(token0, token1) = tokenA < tokenB ? (tokenA, tokenB) : (tokenB, tokenA);
require(token0 != address(0), 'RARESwapLibrary: ZERO_ADDRESS');
}
function pairFor(address factory, address tokenA, address tokenB) internal pure returns (address pair) {
(address token0, address token1) = sortTokens(tokenA, tokenB);
pair = address(uint(keccak256(abi.encodePacked(
hex'ff',
factory,
keccak256(abi.encodePacked(token0, token1)),
hex'8d5eb48a29e28c3f8b5e9069d5c3e5b0913896e4ff564c3a3cef172ae085cfda'
))));
}
function getReserves(address factory, address tokenA, address tokenB) internal view returns (uint reserveA, uint reserveB) {
(address token0,) = sortTokens(tokenA, tokenB);
(uint reserve0, uint reserve1,,) = IRARESwapPair(pairFor(factory, tokenA, tokenB)).getReserves();
(reserveA, reserveB) = tokenA == token0 ? (reserve0, reserve1) : (reserve1, reserve0);
}
function quote(uint amountA, uint reserveA, uint reserveB) internal pure returns (uint amountB) {
require(amountA > 0, 'RARESwapLibrary: INSUFFICIENT_AMOUNT');
require(reserveA > 0 && reserveB > 0, 'RARESwapLibrary: INSUFFICIENT_LIQUIDITY');
amountB = amountA.mul(reserveB) / reserveA;
}
function getAmountOut(uint amountIn, uint reserveIn, uint reserveOut, bool tokenFee, uint totalFee) internal pure returns (uint amountOut) {
require(amountIn > 0, 'RARESwapLibrary: INSUFFICIENT_INPUT_AMOUNT');
require(reserveIn > 0 && reserveOut > 0, 'RARESwapLibrary: INSUFFICIENT_LIQUIDITY');
uint amountInMultiplier = tokenFee ? 10000 - totalFee : 10000;
uint amountInWithFee = amountIn.mul(amountInMultiplier);
uint numerator = amountInWithFee.mul(reserveOut);
uint denominator = reserveIn.mul(10000).add(amountInWithFee);
amountOut = numerator / denominator;
}
function getAmountIn(uint amountOut, uint reserveIn, uint reserveOut, bool tokenFee, uint totalFee) internal pure returns (uint amountIn) {
require(amountOut > 0, 'RARESwapLibrary: INSUFFICIENT_OUTPUT_AMOUNT');
require(reserveIn > 0 && reserveOut > 0, 'RARESwapLibrary: INSUFFICIENT_LIQUIDITY');
uint amountOutMultiplier = tokenFee ? 10000 - totalFee : 10000;
uint numerator = reserveIn.mul(amountOut).mul(10000);
uint denominator = reserveOut.sub(amountOut).mul(amountOutMultiplier);
amountIn = (numerator / denominator).add(1);
}
function getAmountsOut(address factory, uint amountIn, address[] memory path) internal view returns (uint[] memory amounts) {
require(path.length >= 2, 'RARESwapLibrary: INVALID_PATH');
amounts = new uint[](path.length);
amounts[0] = amountIn;
for (uint i; i < path.length - 1; i++) {
IRARESwapPair pair = IRARESwapPair(pairFor(factory, path[i], path[i + 1]));
address baseToken = pair.baseToken();
uint totalFee = pair.getTotalFee();
(uint reserveIn, uint reserveOut) = getReserves(factory, path[i], path[i + 1]);
amounts[i + 1] = getAmountOut(amounts[i], reserveIn, reserveOut, baseToken != address(0), totalFee);
}
}
function getAmountsIn(address factory, uint amountOut, address[] memory path) internal view returns (uint[] memory amounts) {
require(path.length >= 2, 'RARESwapLibrary: INVALID_PATH');
amounts = new uint[](path.length);
amounts[amounts.length - 1] = amountOut;
for (uint i = path.length - 1; i > 0; i--) {
IRARESwapPair pair = IRARESwapPair(pairFor(factory, path[i - 1], path[i]));
address baseToken = pair.baseToken();
uint totalFee = pair.getTotalFee();
(uint reserveIn, uint reserveOut) = getReserves(factory, path[i - 1], path[i]);
amounts[i - 1] = getAmountIn(amounts[i], reserveIn, reserveOut, baseToken != address(0), totalFee);
}
}
function adminFeeCalculation(uint256 _amounts,uint256 _adminFee) internal pure returns (uint256,uint256) {
uint adminFeeDeduct = (_amounts.mul(_adminFee)) / (10000);
_amounts = _amounts.sub(adminFeeDeduct);
return (_amounts,adminFeeDeduct);
}
}
pragma solidity >=0.6.0;
library TransferHelper {
function safeApprove(
address token,
address to,
uint256 value
) internal {
(bool success, bytes memory data) = token.call(abi.encodeWithSelector(0x095ea7b3, to, value));
require(
success && (data.length == 0 || abi.decode(data, (bool))),
'TransferHelper::safeApprove: approve failed'
);
}
function safeTransfer(
address token,
address to,
uint256 value
) internal {
(bool success, bytes memory data) = token.call(abi.encodeWithSelector(0xa9059cbb, to, value));
require(
success && (data.length == 0 || abi.decode(data, (bool))),
'TransferHelper::safeTransfer: transfer failed'
);
}
function safeTransferFrom(
address token,
address from,
address to,
uint256 value
) internal {
(bool success, bytes memory data) = token.call(abi.encodeWithSelector(0x23b872dd, from, to, value));
require(
success && (data.length == 0 || abi.decode(data, (bool))),
'TransferHelper::transferFrom: transferFrom failed'
);
}
function safeTransferETH(address to, uint256 value) internal {
(bool success, ) = to.call{value: value}(new bytes(0));
require(success, 'TransferHelper::safeTransferETH: ETH transfer failed');
}
}
pragma solidity =0.6.6;
abstract contract SupportingSwap is FeeStore, IRARESwapRouter {
using SafeMath for uint;
address public override factory;
address public override WETH;
modifier ensure(uint deadline) {
require(deadline >= block.timestamp, 'RARESwapRouter: EXPIRED');
_;
}
function _swap(uint[] memory amounts, address[] memory path, address _to) internal virtual {
for (uint i; i < path.length - 1; i++) {
(address input, address output) = (path[i], path[i + 1]);
(address token0,) = RARESwapLibrary.sortTokens(input, output);
uint amountOut = amounts[i + 1];
(uint amount0Out, uint amount1Out) = input == token0 ? (uint(0), amountOut) : (amountOut, uint(0));
(uint amount0Fee, uint amount1Fee) = _calculateFees(input, output, amounts[i], amount0Out, amount1Out);
address to = i < path.length - 2 ? RARESwapLibrary.pairFor(factory, output, path[i + 2]) : _to;
IRARESwapPair(RARESwapLibrary.pairFor(factory, input, output)).swap(
amount0Out, amount1Out, amount0Fee, amount1Fee, to, new bytes(0)
);
}
}
function swapExactTokensForTokens(
uint amountIn,
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external virtual override ensure(deadline) returns (uint[] memory amounts) {
require(path.length == 2, "RARESwapRouter: ONLY_TWO_TOKENS_ALLOWED");
address pair = RARESwapLibrary.pairFor(factory, path[0], path[1]);
uint adminFeeDeduct;
if(path[0] == pairFeeAddress[pair]){
(amountIn,adminFeeDeduct) = RARESwapLibrary.adminFeeCalculation(amountIn, adminFee);
TransferHelper.safeTransferFrom(
path[0], msg.sender, feeAddressGet(), adminFeeDeduct
);
}
amounts = RARESwapLibrary.getAmountsOut(factory, amountIn, path);
require(amounts[amounts.length - 1] >= amountOutMin, 'RARESwapRouter: INSUFFICIENT_OUTPUT_AMOUNT');
TransferHelper.safeTransferFrom(
path[0], msg.sender, pair, amounts[0]
);
_swap(amounts, path, to);
}
function swapTokensForExactTokens(
uint amountOut,
uint amountInMax,
address[] calldata path,
address to,
uint deadline
) external virtual override ensure(deadline) returns (uint[] memory amounts) {
require(path.length == 2, "RARESwapRouter: ONLY_TWO_TOKENS_ALLOWED");
address pair = RARESwapLibrary.pairFor(factory, path[0], path[1]);
uint adminFeeDeduct;
if(path[0] == pairFeeAddress[pair]) {
amounts = RARESwapLibrary.getAmountsIn(factory, amountOut, path);
require(amounts[0] <= amountInMax, 'RARESwapRouter: EXCESSIVE_INPUT_AMOUNT');
(amounts[0], adminFeeDeduct) = RARESwapLibrary.adminFeeCalculation(amounts[0], adminFee);
TransferHelper.safeTransferFrom(
path[0], msg.sender, feeAddressGet(), adminFeeDeduct
);
amounts = RARESwapLibrary.getAmountsOut(factory, amounts[0], path);
TransferHelper.safeTransferFrom(
path[0], msg.sender, pair, amounts[0]
);
} else {
amounts = RARESwapLibrary.getAmountsIn(factory, amountOut, path);
require(amounts[0] <= amountInMax, 'RARESwapRouter: EXCESSIVE_INPUT_AMOUNT');
TransferHelper.safeTransferFrom(
path[0], msg.sender, pair, amounts[0]
);
}
_swap(amounts, path, to);
}
function swapExactETHForTokens(uint amountOutMin, address[] calldata path, address to, uint deadline)
external
virtual
override
payable
ensure(deadline)
returns (uint[] memory amounts)
{
require(path.length == 2, "RARESwapRouter: ONLY_TWO_TOKENS_ALLOWED");
require(path[0] == WETH, 'RARESwapRouter: INVALID_PATH');
uint bnb = msg.value;
address pair = RARESwapLibrary.pairFor(factory, path[0], path[1]);
uint adminFeeDeduct;
if(path[0] == pairFeeAddress[pair]){
(bnb, adminFeeDeduct) = RARESwapLibrary.adminFeeCalculation(bnb, adminFee);
if(address(this) != feeAddressGet()){
payable(feeAddressGet()).transfer(adminFeeDeduct);
}
}
amounts = RARESwapLibrary.getAmountsOut(factory, msg.value, path);
require(amounts[amounts.length - 1] >= amountOutMin, 'RARESwapRouter: INSUFFICIENT_OUTPUT_AMOUNT');
IWETH(WETH).deposit{value: amounts[0]}();
assert(IWETH(WETH).transfer(pair, amounts[0]));
_swap(amounts, path, to);
}
function swapTokensForExactETH(uint amountOut, uint amountInMax, address[] calldata path, address to, uint deadline)
external
virtual
override
ensure(deadline)
returns (uint[] memory amounts)
{
require(path.length == 2, "RARESwapRouter: ONLY_TWO_TOKENS_ALLOWED");
require(path[path.length - 1] == WETH, 'RARESwapRouter: INVALID_PATH');
uint adminFeeDeduct;
address pair = RARESwapLibrary.pairFor(factory, path[0], path[1]);
if(path[0] == pairFeeAddress[pair]){
amounts = RARESwapLibrary.getAmountsIn(factory, amountOut, path);
require(amounts[0] <= amountInMax, 'RARESwapRouter: EXCESSIVE_INPUT_AMOUNT');
(amounts[0],adminFeeDeduct) = RARESwapLibrary.adminFeeCalculation(amounts[0],adminFee);
TransferHelper.safeTransferFrom(
path[0], msg.sender, feeAddressGet(), adminFeeDeduct
);
amounts = RARESwapLibrary.getAmountsOut(factory, amounts[0], path);
TransferHelper.safeTransferFrom(
path[0], msg.sender, pair, amounts[0]
);
} else {
amounts = RARESwapLibrary.getAmountsIn(factory, amountOut, path);
require(amounts[0] <= amountInMax, 'RARESwapRouter: EXCESSIVE_INPUT_AMOUNT');
TransferHelper.safeTransferFrom(
path[0], msg.sender, pair, amounts[0]
);
}
_swap(amounts, path, address(this));
uint amountETHOut = amounts[amounts.length - 1];
if(path[1] == pairFeeAddress[pair]){
(amountETHOut,adminFeeDeduct) = RARESwapLibrary.adminFeeCalculation(amountETHOut,adminFee);
}
IWETH(WETH).withdraw(amountETHOut);
TransferHelper.safeTransferETH(to, amountETHOut);
}
function swapExactTokensForETH(uint amountIn, uint amountOutMin, address[] calldata path, address to, uint deadline)
external
virtual
override
ensure(deadline)
returns (uint[] memory amounts)
{
require(path.length == 2, "RARESwapRouter: ONLY_TWO_TOKENS_ALLOWED");
require(path[path.length - 1] == WETH, 'RARESwapRouter: INVALID_PATH');
uint adminFeeDeduct;
address pair = RARESwapLibrary.pairFor(factory, path[0], path[1]);
if(path[0] == pairFeeAddress[pair]){
(amountIn,adminFeeDeduct) = RARESwapLibrary.adminFeeCalculation(amountIn, adminFee);
TransferHelper.safeTransferFrom(
path[0], msg.sender, feeAddressGet(), adminFeeDeduct
);
}
amounts = RARESwapLibrary.getAmountsOut(factory, amountIn, path);
require(amounts[amounts.length - 1] >= amountOutMin, 'RARESwapRouter: INSUFFICIENT_OUTPUT_AMOUNT');
TransferHelper.safeTransferFrom(
path[0], msg.sender, pair, amounts[0]
);
_swap(amounts, path, address(this));
uint amountETHOut = amounts[amounts.length - 1];
if(path[1] == pairFeeAddress[pair]){
(amountETHOut,adminFeeDeduct) = RARESwapLibrary.adminFeeCalculation(amountETHOut,adminFee);
}
IWETH(WETH).withdraw(amountETHOut);
TransferHelper.safeTransferETH(to, amountETHOut);
}
function swapETHForExactTokens(uint amountOut, address[] calldata path, address to, uint deadline)
external
virtual
override
payable
ensure(deadline)
returns (uint[] memory amounts)
{
require(path.length == 2, "RARESwapRouter: ONLY_TWO_TOKENS_ALLOWED");
require(path[0] == WETH, 'RARESwapRouter: INVALID_PATH');
address pair = RARESwapLibrary.pairFor(factory, path[0], path[1]);
uint adminFeeDeduct;
if(path[0] == pairFeeAddress[pair]){
amounts = RARESwapLibrary.getAmountsIn(factory, amountOut, path);
require(amounts[0] <= msg.value, 'RARESwapRouter: EXCESSIVE_INPUT_AMOUNT');
(amounts[0], adminFeeDeduct) = RARESwapLibrary.adminFeeCalculation(amounts[0], adminFee);
if(address(this) != feeAddressGet()){
payable(feeAddressGet()).transfer(adminFeeDeduct);
}
amounts = RARESwapLibrary.getAmountsOut(factory, amounts[0], path);
IWETH(WETH).deposit{value: amounts[0]}();
assert(IWETH(WETH).transfer(pair, amounts[0]));
} else {
amounts = RARESwapLibrary.getAmountsIn(factory, amountOut, path);
require(amounts[0] <= msg.value, 'RARESwapRouter: EXCESSIVE_INPUT_AMOUNT');
IWETH(WETH).deposit{value: amounts[0]}();
assert(IWETH(WETH).transfer(RARESwapLibrary.pairFor(factory, path[0], path[1]), amounts[0]));
}
_swap(amounts, path, to);
uint bal = amounts[0].add(adminFeeDeduct);
if (msg.value > bal) TransferHelper.safeTransferETH(msg.sender, msg.value - bal);
}
function _calculateFees(address input, address output, uint amountIn, uint amount0Out, uint amount1Out) internal view virtual returns (uint amount0Fee, uint amount1Fee) {
IRARESwapPair pair = IRARESwapPair(RARESwapLibrary.pairFor(factory, input, output));
(address token0,) = RARESwapLibrary.sortTokens(input, output);
address baseToken = pair.baseToken();
uint totalFee = pair.getTotalFee();
amount0Fee = baseToken != token0 ? uint(0) : input == token0 ? amountIn.mul(totalFee).div(10**4) : amount0Out.mul(totalFee).div(10**4);
amount1Fee = baseToken == token0 ? uint(0) : input != token0 ? amountIn.mul(totalFee).div(10**4) : amount1Out.mul(totalFee).div(10**4);
}
function _calculateAmounts(address input, address output, address token0) internal view returns (uint amountInput, uint amountOutput) {
IRARESwapPair pair = IRARESwapPair(RARESwapLibrary.pairFor(factory, input, output));
(uint reserve0, uint reserve1,, address baseToken) = pair.getReserves();
uint totalFee = pair.getTotalFee();
(uint reserveInput, uint reserveOutput) = input == token0 ? (reserve0, reserve1) : (reserve1, reserve0);
amountInput = IERC20(input).balanceOf(address(pair)).sub(reserveInput);
amountOutput = RARESwapLibrary.getAmountOut(amountInput, reserveInput, reserveOutput, baseToken != address(0), totalFee);
}
function _swapSupportingFeeOnTransferTokens(address[] memory path, address _to) internal virtual {
for (uint i; i < path.length - 1; i++) {
(address input, address output) = (path[i], path[i + 1]);
(address token0,) = RARESwapLibrary.sortTokens(input, output);
(uint amountInput, uint amountOutput) = _calculateAmounts(input, output, token0);
(uint amount0Out, uint amount1Out) = input == token0 ? (uint(0), amountOutput) : (amountOutput, uint(0));
(uint amount0Fee, uint amount1Fee) = _calculateFees(input, output, amountInput, amount0Out, amount1Out);
address to = i < path.length - 2 ? RARESwapLibrary.pairFor(factory, output, path[i + 2]) : _to;
IRARESwapPair pair = IRARESwapPair(RARESwapLibrary.pairFor(factory, input, output));
pair.swap(amount0Out, amount1Out, amount0Fee, amount1Fee, to, new bytes(0));
}
}
function swapExactTokensForTokensSupportingFeeOnTransferTokens(
uint amountIn,
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external virtual override ensure(deadline) {
require(path.length == 2, "RARESwapRouter: ONLY_TWO_TOKENS_ALLOWED");
address pair = RARESwapLibrary.pairFor(factory, path[0], path[1]);
uint adminFeeDeduct;
if(path[0] == pairFeeAddress[pair]){
(amountIn,adminFeeDeduct) = RARESwapLibrary.adminFeeCalculation(amountIn,adminFee);
TransferHelper.safeTransferFrom(
path[0], msg.sender, feeAddressGet(), adminFeeDeduct
);
}
TransferHelper.safeTransferFrom(
path[0], msg.sender, pair, amountIn
);
uint balanceBefore = IERC20(path[path.length - 1]).balanceOf(to);
_swapSupportingFeeOnTransferTokens(path, to);
if(path[1] == pairFeeAddress[pair]){
(amountOutMin,adminFeeDeduct) = RARESwapLibrary.adminFeeCalculation(amountOutMin,adminFee);
}
require(
IERC20(path[path.length - 1]).balanceOf(to).sub(balanceBefore) >= amountOutMin,
'RARESwapRouter: INSUFFICIENT_OUTPUT_AMOUNT'
);
}
function swapExactETHForTokensSupportingFeeOnTransferTokens(
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
)
external
virtual
override
payable
ensure(deadline)
{
require(path.length == 2, "RARESwapRouter: ONLY_TWO_TOKENS_ALLOWED");
require(path[0] == WETH, 'RARESwapRouter: INVALID_PATH');
uint amountIn = msg.value;
address pair = RARESwapLibrary.pairFor(factory, path[0], path[1]);
uint adminFeeDeduct;
if(path[0] == pairFeeAddress[pair]){
(amountIn,adminFeeDeduct) = RARESwapLibrary.adminFeeCalculation(amountIn,adminFee);
if(address(this) != feeAddressGet()){
payable(feeAddressGet()).transfer(adminFeeDeduct);
}
}
IWETH(WETH).deposit{value: amountIn}();
assert(IWETH(WETH).transfer(pair, amountIn));
uint balanceBefore = IERC20(path[path.length - 1]).balanceOf(to);
_swapSupportingFeeOnTransferTokens(path, to);
if(path[1] == pairFeeAddress[pair]){
(amountOutMin,adminFeeDeduct) = RARESwapLibrary.adminFeeCalculation(amountOutMin,adminFee);
}
require(
IERC20(path[path.length - 1]).balanceOf(to).sub(balanceBefore) >= amountOutMin,
'RARESwapRouter: INSUFFICIENT_OUTPUT_AMOUNT'
);
}
function swapExactTokensForETHSupportingFeeOnTransferTokens(
uint amountIn,
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
)
external
virtual
override
ensure(deadline)
{
require(path.length == 2, "RARESwapRouter: ONLY_TWO_TOKENS_ALLOWED");
require(path[path.length - 1] == WETH, 'RARESwapRouter: INVALID_PATH');
address pair = RARESwapLibrary.pairFor(factory, path[0], path[1]);
if(path[0] == pairFeeAddress[pair]){
uint adminFeeDeduct = (amountIn.mul(adminFee)) / (10000);
amountIn = amountIn.sub(adminFeeDeduct);
TransferHelper.safeTransferFrom(
path[0], msg.sender, feeAddressGet(), adminFeeDeduct
);
}
TransferHelper.safeTransferFrom(
path[0], msg.sender, pair, amountIn
);
_swapSupportingFeeOnTransferTokens(path, address(this));
uint amountOut = IERC20(WETH).balanceOf(address(this));
amountOutMin;
if(path[1] == pairFeeAddress[pair]){
uint adminFeeDeduct = (amountOut.mul(adminFee)) / (10000);
amountOut = amountOut.sub(adminFeeDeduct);
}
IWETH(WETH).withdraw(amountOut);
TransferHelper.safeTransferETH(to, amountOut);
}
function quote(uint amountA, uint reserveA, uint reserveB) public pure virtual override returns (uint amountB) {
return RARESwapLibrary.quote(amountA, reserveA, reserveB);
}
function getAmountOut(uint amountIn, uint reserveIn, uint reserveOut)
public
pure
virtual
override
returns (uint amountOut)
{
return RARESwapLibrary.getAmountOut(amountIn, reserveIn, reserveOut, false, 0);
}
function getAmountIn(uint amountOut, uint reserveIn, uint reserveOut)
public
pure
virtual
override
returns (uint amountIn)
{
return RARESwapLibrary.getAmountIn(amountOut, reserveIn, reserveOut, false, 0);
}
function getAmountsOut(uint amountIn, address[] memory path)
public
view
virtual
override
returns (uint[] memory amounts)
{
return RARESwapLibrary.getAmountsOut(factory, amountIn, path);
}
function getAmountsIn(uint amountOut, address[] memory path)
public
view
virtual
override
returns (uint[] memory amounts)
{
return RARESwapLibrary.getAmountsIn(factory, amountOut, path);
}
}
pragma solidity =0.6.6;
contract RARESwapRouter is SupportingSwap {
using SafeMath for uint;
address private BUSD;
constructor(address _factory, address _WETH, address _BUSD, uint256 _adminFee, address _adminFeeAddress) public {
factory = _factory;
WETH = _WETH;
BUSD = _BUSD;
initialize(_factory, _adminFee, _adminFeeAddress);
IRARESwapFactory(_factory).routerInitialize(address(this));
}
receive() external payable {
assert(msg.sender == WETH);
}
function _addLiquidity(
address tokenA,
address tokenB,
uint amountADesired,
uint amountBDesired,
uint amountAMin,
uint amountBMin
) internal virtual returns (uint amountA, uint amountB) {
if (getPair(tokenA, tokenB) == address(0)) {
if(tokenA == WETH) {
IRARESwapFactory(factory).createPair(tokenB, tokenA);
pairFeeAddress[getPair(tokenA,tokenB)] = tokenA;
} else {
IRARESwapFactory(factory).createPair(tokenA, tokenB);
pairFeeAddress[getPair(tokenA,tokenB)] = tokenB;
}
}
(uint reserveA, uint reserveB) = RARESwapLibrary.getReserves(factory, tokenA, tokenB);
if (reserveA == 0 && reserveB == 0) {
(amountA, amountB) = (amountADesired, amountBDesired);
if (tokenA == WETH) {
pairFeeAddress[getPair(tokenA,tokenB)] = tokenA;
} else if (tokenA == BUSD) {
pairFeeAddress[getPair(tokenA,tokenB)] = tokenA;
} else {
pairFeeAddress[getPair(tokenA,tokenB)] = tokenB;
}
} else {
uint amountBOptimal = RARESwapLibrary.quote(amountADesired, reserveA, reserveB);
if (amountBOptimal <= amountBDesired) {
require(amountBOptimal >= amountBMin, 'RARESwapRouter: INSUFFICIENT_B_AMOUNT');
(amountA, amountB) = (amountADesired, amountBOptimal);
} else {
uint amountAOptimal = RARESwapLibrary.quote(amountBDesired, reserveB, reserveA);
assert(amountAOptimal <= amountADesired);
require(amountAOptimal >= amountAMin, 'RARESwapRouter: INSUFFICIENT_A_AMOUNT');
(amountA, amountB) = (amountAOptimal, amountBDesired);
}
}
}
function getPair(address tokenA,address tokenB) public view returns (address){
return IRARESwapFactory(factory).getPair(tokenA, tokenB);
}
function addLiquidity(
address tokenA,
address tokenB,
uint amountADesired,
uint amountBDesired,
uint amountAMin,
uint amountBMin,
address to,
uint deadline
) external virtual override ensure(deadline) returns (uint amountA, uint amountB, uint liquidity) {
(amountA, amountB) = _addLiquidity(tokenA, tokenB, amountADesired, amountBDesired, amountAMin, amountBMin);
address pair = RARESwapLibrary.pairFor(factory, tokenA, tokenB);
TransferHelper.safeTransferFrom(tokenA, msg.sender, pair, amountA);
TransferHelper.safeTransferFrom(tokenB, msg.sender, pair, amountB);
liquidity = IRARESwapPair(pair).mint(to);
}
function addLiquidityETH(
address token,
uint amountTokenDesired,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline
) external virtual override payable ensure(deadline) returns (uint amountETH, uint amountToken, uint liquidity) {
(amountETH, amountToken) = _addLiquidity(
WETH,
token,
msg.value,
amountTokenDesired,
amountETHMin,
amountTokenMin
);
address pair = RARESwapLibrary.pairFor(factory, token, WETH);
TransferHelper.safeTransferFrom(token, msg.sender, pair, amountToken);
IWETH(WETH).deposit{value: amountETH}();
assert(IWETH(WETH).transfer(pair, amountETH));
liquidity = IRARESwapPair(pair).mint(to);
if (msg.value > amountETH) TransferHelper.safeTransferETH(msg.sender, msg.value - amountETH);
}
function removeLiquidity(
address tokenA,
address tokenB,
uint liquidity,
uint amountAMin,
uint amountBMin,
address to,
uint deadline
) public virtual override ensure(deadline) returns (uint amountA, uint amountB) {
address pair = RARESwapLibrary.pairFor(factory, tokenA, tokenB);
IRARESwapPair(pair).transferFrom(msg.sender, pair, liquidity);
(uint amount0, uint amount1) = IRARESwapPair(pair).burn(to);
(address token0,) = RARESwapLibrary.sortTokens(tokenA, tokenB);
(amountA, amountB) = tokenA == token0 ? (amount0, amount1) : (amount1, amount0);
require(amountA >= amountAMin, 'RARESwapRouter: INSUFFICIENT_A_AMOUNT');
require(amountB >= amountBMin, 'RARESwapRouter: INSUFFICIENT_B_AMOUNT');
}
function removeLiquidityETH(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline
) public virtual override ensure(deadline) returns (uint amountToken, uint amountETH) {
(amountToken, amountETH) = removeLiquidity(
token,
WETH,
liquidity,
amountTokenMin,
amountETHMin,
address(this),
deadline
);
TransferHelper.safeTransfer(token, to, amountToken);
IWETH(WETH).withdraw(amountETH);
TransferHelper.safeTransferETH(to, amountETH);
}
function removeLiquidityWithPermit(
address tokenA,
address tokenB,
uint liquidity,
uint amountAMin,
uint amountBMin,
address to,
uint deadline,
bool approveMax, uint8 v, bytes32 r, bytes32 s
) external virtual override returns (uint amountA, uint amountB) {
address pair = RARESwapLibrary.pairFor(factory, tokenA, tokenB);
uint value = approveMax ? uint(-1) : liquidity;
IRARESwapPair(pair).permit(msg.sender, address(this), value, deadline, v, r, s);
(amountA, amountB) = removeLiquidity(tokenA, tokenB, liquidity, amountAMin, amountBMin, to, deadline);
}
function removeLiquidityETHWithPermit(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline,
bool approveMax, uint8 v, bytes32 r, bytes32 s
) external virtual override returns (uint amountToken, uint amountETH) {
address pair = RARESwapLibrary.pairFor(factory, token, WETH);
uint value = approveMax ? uint(-1) : liquidity;
IRARESwapPair(pair).permit(msg.sender, address(this), value, deadline, v, r, s);
(amountToken, amountETH) = removeLiquidityETH(token, liquidity, amountTokenMin, amountETHMin, to, deadline);
}
function removeLiquidityETHSupportingFeeOnTransferTokens(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline
) public virtual override ensure(deadline) returns (uint amountETH) {
(, amountETH) = removeLiquidity(
token,
WETH,
liquidity,
amountTokenMin,
amountETHMin,
address(this),
deadline
);
TransferHelper.safeTransfer(token, to, IERC20(token).balanceOf(address(this)));
IWETH(WETH).withdraw(amountETH);
TransferHelper.safeTransferETH(to, amountETH);
}
function removeLiquidityETHWithPermitSupportingFeeOnTransferTokens(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline,
bool approveMax, uint8 v, bytes32 r, bytes32 s
) external virtual override returns (uint amountETH) {
address pair = RARESwapLibrary.pairFor(factory, token, WETH);
uint value = approveMax ? uint(-1) : liquidity;
IRARESwapPair(pair).permit(msg.sender, address(this), value, deadline, v, r, s);
amountETH = removeLiquidityETHSupportingFeeOnTransferTokens(
token, liquidity, amountTokenMin, amountETHMin, to, deadline
);
}
}