// SPDX-License-Identifier: GPL-3.0 pragma solidity =0.6.12; import '../interfaces/IFeSwapPair.sol'; import '../interfaces/IFeSwapFactory.sol'; import './TransferHelper.sol'; import "./SafeMath.sol"; library FeSwapLibrary { using SafeMath for uint; // calculates the CREATE2 address for a pair without making any external calls function pairFor(address factory, address tokenA, address tokenB) internal pure returns (address pair) { pair = address(uint(keccak256(abi.encodePacked( hex'ff', factory, keccak256(abi.encodePacked(tokenA, tokenB)), hex'baeae6b526d52ab9edea88a8a5991ed61443a0c4d6bea03dc99971664961f4b7' // init code hash // save 9916 gas )))); } // fetches and sorts the reserves for a pair function getReserves(address factory, address tokenA, address tokenB) internal view returns (uint reserveA, uint reserveB, address pair, uint rateTriggerArbitrage) { pair = pairFor(factory, tokenA, tokenB); (reserveA, reserveB, , rateTriggerArbitrage) = IFeSwapPair(pair).getReserves(); } // given some amount of an asset and pair reserves, returns an equivalent amount of the other asset function quote(uint amountA, uint reserveA, uint reserveB) internal pure returns (uint amountB) { require(amountA > 0, 'FeSwapLibrary: INSUFFICIENT_AMOUNT'); require(reserveA > 0 && reserveB > 0, 'FeSwapLibrary: INSUFFICIENT_LIQUIDITY'); amountB = amountA.mul(reserveB) / reserveA; } // given an input amount of an asset and pair reserves, returns the maximum output amount of the other asset function getAmountOut(uint amountIn, uint reserveIn, uint reserveOut) internal pure returns (uint amountOut) { require(amountIn > 0, 'FeSwapLibrary: INSUFFICIENT_INPUT_AMOUNT'); require(reserveIn > 0 && reserveOut > 0, 'FeSwapLibrary: INSUFFICIENT_LIQUIDITY'); uint numerator = amountIn.mul(reserveOut); uint denominator = reserveIn.add(amountIn); amountOut = numerator / denominator; } // given an output amount of an asset and pair reserves, returns a required input amount of the other asset function getAmountIn(uint amountOut, uint reserveIn, uint reserveOut) internal pure returns (uint amountIn) { require(amountOut > 0, 'FeSwapLibrary: INSUFFICIENT_OUTPUT_AMOUNT'); require(reserveIn > 0 && reserveOut > 0, 'FeSwapLibrary: INSUFFICIENT_LIQUIDITY'); uint numerator = reserveIn.mul(amountOut); uint denominator = reserveOut.sub(amountOut); // amountIn = (numerator / denominator).add(1); amountIn = (numerator.add(denominator)) / denominator; } function arbitragePairPools(address factory, address tokenA, address tokenB) internal returns (uint reserveIn, uint reserveOut, address pair) { (reserveIn, reserveOut, pair, ) = getReserves(factory, tokenA, tokenB); (uint reserveInMate, uint reserveOutMate, address PairMate, uint rateTriggerArbitrage) = FeSwapLibrary.getReserves(factory, tokenB, tokenA); uint productIn = uint(reserveIn).mul(reserveInMate); uint productOut = uint(reserveOut).mul(reserveOutMate); if(productIn.mul(10000) > productOut.mul(rateTriggerArbitrage)){ uint ArbitatrageIn = productIn.sub(productOut); uint totalTokenA = (uint(reserveIn).add(reserveOutMate)).mul(2); uint totalTokenB = (uint(reserveOut).add(reserveInMate)).mul(2); TransferHelper.safeTransferFrom(tokenA, pair, PairMate, ArbitatrageIn / totalTokenB); TransferHelper.safeTransferFrom(tokenB, PairMate, pair, ArbitatrageIn / totalTokenA); IFeSwapPair(pair).sync(); IFeSwapPair(PairMate).sync(); (reserveIn, reserveOut, ,) = getReserves(factory, tokenA, tokenB); } } // performs chained getAmountOut calculations on any number of pairs function getAmountsOut(address factory, uint amountIn, address[] calldata path) internal returns (address firstPair, uint[] memory amounts) { require(path.length >= 2, 'FeSwapLibrary: INVALID_PATH'); amounts = new uint[](path.length); amounts[0] = amountIn; for (uint i = 0; i < path.length - 1; i++) { (uint reserveIn, uint reserveOut, address _firstPair) = arbitragePairPools(factory, path[i], path[i + 1]); amounts[i + 1] = getAmountOut(amounts[i], reserveIn, reserveOut); if ( i == 0 ) firstPair = _firstPair; } } // performs chained estimateAmountsOut calculations on any number of pairs function estimateAmountsOut(address factory, uint amountIn, address[] calldata path) internal view returns (uint[] memory amounts) { require(path.length >= 2, 'FeSwapLibrary: INVALID_PATH'); amounts = new uint[](path.length); amounts[0] = amountIn; for (uint i = 0; i < path.length - 1; i++) { (uint reserveIn, uint reserveOut, , ) = getReserves(factory, path[i], path[i + 1]); amounts[i + 1] = getAmountOut(amounts[i], reserveIn, reserveOut); } } // performs chained getAmountIn calculations on any number of pairs function getAmountsIn(address factory, uint amountOut, address[] calldata path) internal returns (address firstPair, uint[] memory amounts) { require(path.length >= 2, 'FeSwapLibrary: INVALID_PATH'); amounts = new uint[](path.length); amounts[amounts.length - 1] = amountOut; uint reserveIn; uint reserveOut; for (uint i = path.length - 1; i > 0; i--) { (reserveIn, reserveOut, firstPair) = arbitragePairPools(factory, path[i - 1], path[i]); amounts[i - 1] = getAmountIn(amounts[i], reserveIn, reserveOut); } } function estimateAmountsIn(address factory, uint amountOut, address[] calldata path) internal view returns (uint[] memory amounts) { require(path.length >= 2, 'FeSwapLibrary: INVALID_PATH'); amounts = new uint[](path.length); amounts[amounts.length - 1] = amountOut; for (uint i = path.length - 1; i > 0; i--) { (uint reserveIn, uint reserveOut, ,) = getReserves(factory, path[i - 1], path[i]); amounts[i - 1] = getAmountIn(amounts[i], reserveIn, reserveOut); } } }