文件 1 的 17:Address.sol
pragma solidity >=0.6.0 <=0.8.9;
library Address {
function isContract(address account) internal view returns (bool) {
uint256 size;
assembly {
size := extcodesize(account)
}
return size > 0;
}
function sendValue(address payable recipient, uint256 amount) internal {
require(address(this).balance >= amount, "Address: insufficient balance");
(bool success, ) = recipient.call{value: amount}("");
require(success, "Address: unable to send value, recipient may have reverted");
}
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCall(target, data, "Address: low-level call failed");
}
function functionCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value
) internal returns (bytes memory) {
return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
}
function functionCallWithValue(
address target,
bytes memory data,
uint256 value,
string memory errorMessage
) internal returns (bytes memory) {
require(address(this).balance >= value, "Address: insufficient balance for call");
require(isContract(target), "Address: call to non-contract");
(bool success, bytes memory returndata) = target.call{value: value}(data);
return verifyCallResult(success, returndata, errorMessage);
}
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
function functionStaticCall(
address target,
bytes memory data,
string memory errorMessage
) internal view returns (bytes memory) {
require(isContract(target), "Address: static call to non-contract");
(bool success, bytes memory returndata) = target.staticcall(data);
return verifyCallResult(success, returndata, errorMessage);
}
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
function functionDelegateCall(
address target,
bytes memory data,
string memory errorMessage
) internal returns (bytes memory) {
require(isContract(target), "Address: delegate call to non-contract");
(bool success, bytes memory returndata) = target.delegatecall(data);
return verifyCallResult(success, returndata, errorMessage);
}
function verifyCallResult(
bool success,
bytes memory returndata,
string memory errorMessage
) internal pure returns (bytes memory) {
if (success) {
return returndata;
} else {
if (returndata.length > 0) {
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
}
文件 2 的 17:Context.sol
pragma solidity ^0.8.0;
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}
文件 3 的 17:IAugustusSwapper.sol
pragma solidity >=0.6.0 <=0.8.9;
pragma experimental ABIEncoderV2;
import "./IERC20.sol";
interface IAugustusSwapper {
struct SellData {
address fromToken;
uint256 fromAmount;
uint256 toAmount;
uint256 expectedAmount;
address payable beneficiary;
string referrer;
bool useReduxToken;
Path[] path;
}
struct MegaSwapSellData {
address fromToken;
uint256 fromAmount;
uint256 toAmount;
uint256 expectedAmount;
address payable beneficiary;
string referrer;
bool useReduxToken;
MegaSwapPath[] path;
}
struct BuyData {
address fromToken;
address toToken;
uint256 fromAmount;
uint256 toAmount;
address payable beneficiary;
string referrer;
bool useReduxToken;
BuyRoute[] route;
}
struct Route {
address payable exchange;
address targetExchange;
uint percent;
bytes payload;
uint256 networkFee;
}
struct MegaSwapPath {
uint256 fromAmountPercent;
Path[] path;
}
struct Path {
address to;
uint256 totalNetworkFee;
Route[] routes;
}
struct BuyRoute {
address payable exchange;
address targetExchange;
uint256 fromAmount;
uint256 toAmount;
bytes payload;
uint256 networkFee;
}
function getPartnerRegistry() external view returns(address);
function getWhitelistAddress() external view returns(address);
function getFeeWallet() external view returns(address);
function getTokenTransferProxy() external view returns (address);
function getUniswapProxy() external view returns(address);
function getVersion() external view returns(string memory);
function multiSwap(
SellData calldata data
)
external
payable
returns (uint256);
function buy(
BuyData calldata data
)
external
payable
returns (uint256);
function swapOnUniswap(
uint256 amountIn,
uint256 amountOutMin,
address[] calldata path,
uint8 referrer
)
external
payable;
function buyOnUniswap(
uint256 amountInMax,
uint256 amountOut,
address[] calldata path,
uint8 referrer
)
external
payable;
function buyOnUniswapFork(
address factory,
bytes32 initCode,
uint256 amountInMax,
uint256 amountOut,
address[] calldata path,
uint8 referrer
)
external
payable;
function swapOnUniswapFork(
address factory,
bytes32 initCode,
uint256 amountIn,
uint256 amountOutMin,
address[] calldata path,
uint8 referrer
)
external
payable;
function simplBuy(
address fromToken,
address toToken,
uint256 fromAmount,
uint256 toAmount,
address[] memory callees,
bytes memory exchangeData,
uint256[] memory startIndexes,
uint256[] memory values,
address payable beneficiary,
string memory referrer,
bool useReduxToken
)
external
payable;
function simpleSwap(
address fromToken,
address toToken,
uint256 fromAmount,
uint256 toAmount,
uint256 expectedAmount,
address[] memory callees,
bytes memory exchangeData,
uint256[] memory startIndexes,
uint256[] memory values,
address payable beneficiary,
string memory referrer,
bool useReduxToken
)
external
payable
returns (uint256 receivedAmount);
function megaSwap(
MegaSwapSellData memory data
)
external
payable
returns (uint256);
}
文件 4 的 17:IBurnableToken.sol
pragma solidity >=0.6.0 <=0.8.9;
import "./IERC20.sol";
interface IBurnableToken is IERC20 {
function mint(address target, uint256 amount) external returns (bool);
function burn(uint256 amount) external returns (bool);
function mintable() external returns (bool);
}
文件 5 的 17:IERC20.sol
pragma solidity >=0.6.0 <=0.8.9;
interface IERC20 {
function totalSupply() external view returns (uint256);
function decimals() external view returns (uint8);
function symbol() external view returns (string memory);
function name() external view returns (string memory);
function getOwner() external view returns (address);
function balanceOf(address account) external view returns (uint256);
function transfer(address recipient, uint256 amount)
external
returns (bool);
function allowance(address _owner, address spender)
external
view
returns (uint256);
function approve(address spender, uint256 amount) external returns (bool);
function transferFrom(
address sender,
address recipient,
uint256 amount
) external returns (bool);
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(
address indexed owner,
address indexed spender,
uint256 value
);
}
文件 6 的 17:IParams.sol
pragma solidity >=0.6.0 <=0.8.9;
pragma experimental ABIEncoderV2;
interface IParams {
function minimumSwapAmountForWBTC() external view returns (uint256);
function expirationTime() external view returns (uint256);
function paraswapAddress() external view returns (address);
function nodeRewardsRatio() external view returns (uint8);
function depositFeesBPS() external view returns (uint8);
function withdrawalFeeBPS() external view returns (uint8);
function loopCount() external view returns (uint8);
function setMinimumSwapAmountForWBTC(uint256 _minimumSwapAmountForWBTC) external;
function setExpirationTime(uint256 _expirationTime) external;
function setParaswapAddress(address _paraswapAddress) external;
function setNodeRewardsRatio(uint8 _nodeRewardsRatio) external;
function setWithdrawalFeeBPS(uint8 _withdrawalFeeBPS) external;
function setDepositFeesBPS(uint8 _depositFeesBPS) external;
function setLoopCount(uint8 _loopCount) external;
}
文件 7 的 17:IParaswap.sol
pragma solidity >=0.6.0 <=0.8.9;
pragma experimental ABIEncoderV2;
import "./lib/Utils.sol";
import "./IERC20.sol";
interface IParaswap {
function multiSwap(
Utils.SellData calldata data
)
external
payable
returns (uint256);
function megaSwap(
Utils.MegaSwapSellData calldata data
)
external
payable
returns (uint256);
function protectedMultiSwap(
Utils.SellData calldata data
)
external
payable
returns (uint256);
function protectedMegaSwap(
Utils.MegaSwapSellData calldata data
)
external
payable
returns (uint256);
function protectedSimpleSwap(
Utils.SimpleData calldata data
)
external
payable
returns (uint256 receivedAmount);
function protectedSimpleBuy(
Utils.SimpleData calldata data
)
external
payable;
function simpleSwap(
Utils.SimpleData calldata data
)
external
payable
returns (uint256 receivedAmount);
function simpleBuy(
Utils.SimpleData calldata data
)
external
payable;
function swapOnUniswap(
uint256 amountIn,
uint256 amountOutMin,
address[] calldata path
)
external
payable;
function swapOnUniswapFork(
address factory,
bytes32 initCode,
uint256 amountIn,
uint256 amountOutMin,
address[] calldata path
)
external
payable;
function buyOnUniswap(
uint256 amountInMax,
uint256 amountOut,
address[] calldata path
)
external
payable;
function buyOnUniswapFork(
address factory,
bytes32 initCode,
uint256 amountInMax,
uint256 amountOut,
address[] calldata path
)
external
payable;
function swapOnUniswapV2Fork(
address tokenIn,
uint256 amountIn,
uint256 amountOutMin,
address weth,
uint256[] calldata pools
)
external
payable;
function buyOnUniswapV2Fork(
address tokenIn,
uint256 amountInMax,
uint256 amountOut,
address weth,
uint256[] calldata pools
)
external
payable;
function swapOnZeroXv2(
IERC20 fromToken,
IERC20 toToken,
uint256 fromAmount,
uint256 amountOutMin,
address exchange,
bytes calldata payload
)
external
payable;
function swapOnZeroXv4(
IERC20 fromToken,
IERC20 toToken,
uint256 fromAmount,
uint256 amountOutMin,
address exchange,
bytes calldata payload
)
external
payable;
}
文件 8 的 17:ISwapContract.sol
pragma solidity >=0.6.0 <=0.8.9;
pragma experimental ABIEncoderV2;
import "./lib/Utils.sol";
interface ISwapContract {
function BTCT_ADDR() external returns (address);
function singleTransferERC20(
address _destToken,
address _to,
uint256 _amount,
uint256 _totalSwapped,
uint256 _rewardsAmount,
bytes32[] memory _redeemedFloatTxIds
) external returns (bool);
function multiTransferERC20TightlyPacked(
address _destToken,
bytes32[] memory _addressesAndAmounts,
uint256 _totalSwapped,
uint256 _rewardsAmount,
bytes32[] memory _redeemedFloatTxIds
) external returns (bool);
function collectSwapFeesForBTC(
uint256 _incomingAmount,
uint256 _minerFee,
uint256 _rewardsAmount,
address[] memory _spenders,
uint256[] memory _swapAmounts,
bool _isUpdatelimitBTCForSPFlow2
) external returns (bool);
function recordIncomingFloat(
address _token,
bytes32 _addressesAndAmountOfFloat,
bytes32 _txid
) external returns (bool);
function recordOutcomingFloat(
address _token,
bytes32 _addressesAndAmountOfLPtoken,
uint256 _minerFee,
bytes32 _txid
) external returns (bool);
function recordSkyPoolsTX(
address _to,
uint256 _totalSwapped,
uint256 _rewardsAmount,
bytes32[] memory _usedTxIds
) external returns (bool);
function spFlow1SimpleSwap(Utils.SimpleData calldata _data) external;
function spFlow1Uniswap(
bool _fork,
address _factory,
bytes32 _initCode,
uint256 _amountIn,
uint256 _amountOutMin,
address[] calldata _path
) external returns (uint256 receivedAmount);
function spFlow2Uniswap(
string memory _destinationAddressForBTC,
bool _fork,
address _factory,
bytes32 _initCode,
uint256 _amountIn,
uint256 _amountOutMin,
address[] calldata _path
) external returns (uint256 receivedAmount);
function spFlow2SimpleSwap(
string memory _destinationAddressForBTC,
Utils.SimpleData calldata _data
) external returns (uint256 receivedAmount);
function spCleanUpOldTXs() external;
function spDeposit(address _token, uint256 _amount) external payable;
function redeemEther(uint256 _amount) external;
function redeemERC20Token(address _token, uint256 _amount) external;
function recordUTXOSweepMinerFee(uint256 _minerFee, bytes32 _txid)
external
returns (bool);
function churn(
address _newOwner,
address[] memory _nodes,
bool[] memory _isRemoved,
uint8 _churnedInCount,
uint8 _tssThreshold
) external returns (bool);
function isTxUsed(bytes32 _txid) external view returns (bool);
function getCurrentPriceLP() external view returns (uint256);
function getFloatReserve(address _tokenA, address _tokenB)
external
returns (uint256 reserveA, uint256 reserveB);
function getActiveNodes() external view returns (address[] memory);
function isNodeStake(address _user) external returns (bool);
}
文件 9 的 17:ISwapRewards.sol
pragma solidity >=0.6.0 <=0.8.9;
interface ISwapRewards {
function setSWINGBYPrice(uint256 _pricePerBTC) external;
function pullRewards(address _dest, address _receiver, uint256 _swapped) external returns (bool);
function pullRewardsMulti(address _dest, address[] memory _receiver, uint256[] memory _swapped) external returns (bool);
}
文件 10 的 17:ITokenTransferProxy.sol
pragma solidity >=0.6.0 <=0.8.9;
interface ITokenTransferProxy {
function transferFrom(
address token,
address from,
address to,
uint256 amount
)
external;
function freeReduxTokens(address user, uint256 tokensToFree) external;
}
文件 11 的 17:IWETH.sol
pragma solidity >=0.5.0;
interface IWETH {
function deposit() external payable;
function transfer(address to, uint256 value) external returns (bool);
function withdraw(uint256) external;
function approve(address spender, uint256 amount) external returns (bool);
}
文件 12 的 17:Ownable.sol
pragma solidity ^0.8.0;
import "../utils/Context.sol";
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
constructor() {
_setOwner(_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 {
_setOwner(address(0));
}
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
_setOwner(newOwner);
}
function _setOwner(address newOwner) private {
address oldOwner = _owner;
_owner = newOwner;
emit OwnershipTransferred(oldOwner, newOwner);
}
}
文件 13 的 17:ReentrancyGuard.sol
pragma solidity ^0.8.0;
abstract contract ReentrancyGuard {
uint256 private constant _NOT_ENTERED = 1;
uint256 private constant _ENTERED = 2;
uint256 private _status;
constructor() {
_status = _NOT_ENTERED;
}
modifier nonReentrant() {
require(_status != _ENTERED, "ReentrancyGuard: reentrant call");
_status = _ENTERED;
_;
_status = _NOT_ENTERED;
}
}
文件 14 的 17:SafeERC20.sol
pragma solidity >=0.6.0 <=0.8.9;
import "../IERC20.sol";
import "./Address.sol";
library SafeERC20 {
using Address for address;
function safeTransfer(
IERC20 token,
address to,
uint256 value
) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
}
function safeTransferFrom(
IERC20 token,
address from,
address to,
uint256 value
) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
}
function safeApprove(
IERC20 token,
address spender,
uint256 value
) internal {
require(
(value == 0) || (token.allowance(address(this), spender) == 0),
"SafeERC20: approve from non-zero to non-zero allowance"
);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
}
function safeIncreaseAllowance(
IERC20 token,
address spender,
uint256 value
) internal {
uint256 newAllowance = token.allowance(address(this), spender) + value;
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
}
function _callOptionalReturn(IERC20 token, bytes memory data) private {
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
if (returndata.length > 0) {
require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
}
}
文件 15 的 17:SafeMath.sol
pragma solidity ^0.8.0;
library SafeMath {
function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
uint256 c = a + b;
if (c < a) return (false, 0);
return (true, c);
}
}
function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
if (b > a) return (false, 0);
return (true, a - b);
}
}
function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
if (a == 0) return (true, 0);
uint256 c = a * b;
if (c / a != b) return (false, 0);
return (true, c);
}
}
function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
if (b == 0) return (false, 0);
return (true, a / b);
}
}
function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) {
unchecked {
if (b == 0) return (false, 0);
return (true, a % b);
}
}
function add(uint256 a, uint256 b) internal pure returns (uint256) {
return a + b;
}
function sub(uint256 a, uint256 b) internal pure returns (uint256) {
return a - b;
}
function mul(uint256 a, uint256 b) internal pure returns (uint256) {
return a * b;
}
function div(uint256 a, uint256 b) internal pure returns (uint256) {
return a / b;
}
function mod(uint256 a, uint256 b) internal pure returns (uint256) {
return a % b;
}
function sub(
uint256 a,
uint256 b,
string memory errorMessage
) internal pure returns (uint256) {
unchecked {
require(b <= a, errorMessage);
return a - b;
}
}
function div(
uint256 a,
uint256 b,
string memory errorMessage
) internal pure returns (uint256) {
unchecked {
require(b > 0, errorMessage);
return a / b;
}
}
function mod(
uint256 a,
uint256 b,
string memory errorMessage
) internal pure returns (uint256) {
unchecked {
require(b > 0, errorMessage);
return a % b;
}
}
}
文件 16 的 17:SwapContract.sol
pragma solidity >=0.6.0 <=0.8.9;
pragma experimental ABIEncoderV2;
import "./interfaces/IWETH.sol";
import "./interfaces/IBurnableToken.sol";
import "./interfaces/IERC20.sol";
import "./interfaces/ISwapContract.sol";
import "./interfaces/ISwapRewards.sol";
import "./interfaces/IParams.sol";
import "./interfaces/IAugustusSwapper.sol";
import "./interfaces/ITokenTransferProxy.sol";
import "./interfaces/IParaswap.sol";
import "./interfaces/lib/Utils.sol";
import "@openzeppelin/contracts/access/Ownable.sol";
import "./interfaces/lib/SafeERC20.sol";
import "@openzeppelin/contracts/utils/math/SafeMath.sol";
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";
contract SwapContract is Ownable, ReentrancyGuard, ISwapContract {
using SafeMath for uint256;
using SafeERC20 for IERC20;
struct spPendingTx {
bytes32 SwapID;
string DestAddr;
address RefundAddr;
uint256 AmountWBTC;
uint256 Timestamp;
}
IBurnableToken public immutable lpToken;
IParams public immutable ip;
ISwapRewards public immutable sw;
mapping(address => bool) public whitelist;
address public immutable BTCT_ADDR;
address public immutable sbBTCPool;
uint256 private immutable convertScale;
uint256 private immutable lpDivisor;
mapping(address => uint256) private floatAmountOf;
mapping(bytes32 => bool) private used;
mapping(address => uint8) private nodes;
address[] private nodeAddrs;
uint8 public activeNodeCount;
uint8 public churnedInCount;
uint8 public tssThreshold;
mapping(address => mapping(address => uint256)) public tokens;
address constant ETHER = 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE;
address public paraswapAddress;
address public immutable wETH;
mapping(uint256 => spPendingTx) public spPendingTXs;
uint256 public swapCount;
uint256 public oldestActiveIndex;
uint256 public limitBTCForSPFlow2;
event Swap(address from, address to, uint256 amount);
event Withdraw(
address token,
address user,
uint256 amount,
uint256 balance,
uint256 Timestamp
);
event Deposit(
address token,
address user,
uint256 amount,
uint256 balance,
uint256 Timestamp
);
event RewardsCollection(
address feesToken,
uint256 rewards,
uint256 amountLPTokensForNode,
uint256 currentPriceLP
);
event IssueLPTokensForFloat(
address to,
uint256 amountOfFloat,
uint256 amountOfLP,
uint256 currentPriceLP,
uint256 depositFees,
bytes32 txid
);
event BurnLPTokensForFloat(
address token,
uint256 amountOfLP,
uint256 amountOfFloat,
uint256 currentPriceLP,
uint256 withdrawal,
bytes32 txid
);
event SwapTokensToBTC(
bytes32 SwapID,
string DestAddr,
address RefundAddr,
uint256 AmountWBTC,
uint256 Timestamp
);
event DistributeNodeRewards(uint256 rewardLPTsForNodes);
modifier priceCheck() {
uint256 beforePrice = getCurrentPriceLP();
_;
require(getCurrentPriceLP() >= beforePrice, "Invalid LPT price");
}
constructor(
address _lpToken,
address _btct,
address _wETH,
address _sbBTCPool,
address _params,
address _swapRewards,
uint256 _existingBTCFloat
) {
oldestActiveIndex = 0;
swapCount = 0;
wETH = _wETH;
sbBTCPool = _sbBTCPool;
ip = IParams(_params);
paraswapAddress = ip.paraswapAddress();
sw = ISwapRewards(_swapRewards);
lpToken = IBurnableToken(_lpToken);
lpDivisor = 10**IERC20(_lpToken).decimals();
BTCT_ADDR = _btct;
convertScale = 10**(IERC20(_btct).decimals() - 8);
whitelist[_btct] = true;
whitelist[_lpToken] = true;
whitelist[address(0)] = true;
floatAmountOf[address(0)] = _existingBTCFloat;
}
function singleTransferERC20(
address _destToken,
address _to,
uint256 _amount,
uint256 _totalSwapped,
uint256 _rewardsAmount,
bytes32[] memory _redeemedFloatTxIds
) external override onlyOwner returns (bool) {
require(whitelist[_destToken], "14");
require(
_destToken != address(0),
"15"
);
address _feesToken = address(0);
if (_totalSwapped > 0) {
sw.pullRewards(_destToken, _to, _totalSwapped);
_swap(address(0), BTCT_ADDR, _totalSwapped);
} else {
_feesToken = (_destToken == address(lpToken)) ? address(lpToken) : BTCT_ADDR;
}
_rewardsCollection(_feesToken, _rewardsAmount);
_addUsedTxs(_redeemedFloatTxIds);
_safeTransfer(_destToken, _to, _amount);
return true;
}
function multiTransferERC20TightlyPacked(
address _destToken,
bytes32[] memory _addressesAndAmounts,
uint256 _totalSwapped,
uint256 _rewardsAmount,
bytes32[] memory _redeemedFloatTxIds
) external override onlyOwner returns (bool) {
require(whitelist[_destToken], "_destToken is not whitelisted");
require(
_destToken != address(0),
"_destToken should not be address(0)"
);
address _feesToken = address(0);
if (_totalSwapped > 0) {
_swap(address(0), BTCT_ADDR, _totalSwapped);
} else {
_feesToken = (_destToken == address(lpToken)) ? address(lpToken) : BTCT_ADDR;
}
_rewardsCollection(_feesToken, _rewardsAmount);
_addUsedTxs(_redeemedFloatTxIds);
for (uint256 i = 0; i < _addressesAndAmounts.length; i++) {
_safeTransfer(
_destToken,
address(uint160(uint256(_addressesAndAmounts[i]))),
uint256(uint96(bytes12(_addressesAndAmounts[i])))
);
}
return true;
}
function collectSwapFeesForBTC(
uint256 _incomingAmount,
uint256 _minerFee,
uint256 _rewardsAmount,
address[] memory _spenders,
uint256[] memory _swapAmounts,
bool _isUpdatelimitBTCForSPFlow2
) external override onlyOwner returns (bool) {
address _feesToken = BTCT_ADDR;
if (_incomingAmount > 0) {
uint256 swapAmount = _incomingAmount.sub(_rewardsAmount);
sw.pullRewardsMulti(address(0), _spenders, _swapAmounts);
_swap(BTCT_ADDR, address(0), swapAmount);
} else if (_incomingAmount == 0) {
_feesToken = address(0);
}
_rewardsCollection(_feesToken, _rewardsAmount);
if (_isUpdatelimitBTCForSPFlow2) {
_updateLimitBTCForSPFlow2();
}
return true;
}
function recordIncomingFloat(
address _token,
bytes32 _addressesAndAmountOfFloat,
bytes32 _txid
) external override onlyOwner priceCheck returns (bool) {
require(whitelist[_token], "16");
require(
_issueLPTokensForFloat(_token, _addressesAndAmountOfFloat, _txid)
);
return true;
}
function recordOutcomingFloat(
address _token,
bytes32 _addressesAndAmountOfLPtoken,
uint256 _minerFee,
bytes32 _txid
) external override onlyOwner priceCheck returns (bool) {
require(whitelist[_token], "16");
require(
_burnLPTokensForFloat(
_token,
_addressesAndAmountOfLPtoken,
_minerFee,
_txid
)
);
return true;
}
function recordSkyPoolsTX(
address _to,
uint256 _totalSwapped,
uint256 _rewardsAmount,
bytes32[] memory _usedTxIds
) external onlyOwner returns (bool) {
require(_totalSwapped != 0);
require(_rewardsAmount != 0);
_swap(address(0), BTCT_ADDR, _totalSwapped);
tokens[BTCT_ADDR][_to] = tokens[BTCT_ADDR][_to].add(_totalSwapped);
_rewardsCollection(address(0), _rewardsAmount);
_addUsedTxs(_usedTxIds);
return true;
}
function multiRecordSkyPoolsTX(
bytes32[] memory _addressesAndAmounts,
uint256 _totalSwapped,
uint256 _rewardsAmount,
bytes32[] memory _usedTxIds
) external onlyOwner returns (bool) {
require(_totalSwapped != 0);
require(_rewardsAmount != 0);
_swap(address(0), BTCT_ADDR, _totalSwapped);
_rewardsCollection(address(0), _rewardsAmount);
_addUsedTxs(_usedTxIds);
for (uint256 i = 0; i < _addressesAndAmounts.length; i++) {
tokens[BTCT_ADDR][
address(uint160(uint256(_addressesAndAmounts[i])))
] = tokens[BTCT_ADDR][
address(uint160(uint256(_addressesAndAmounts[i])))
].add(uint256(uint96(bytes12(_addressesAndAmounts[i]))));
}
return true;
}
function spFlow1SimpleSwap(Utils.SimpleData calldata _data)
external
nonReentrant
{
require(_data.beneficiary == msg.sender, "beneficiary != msg.sender");
require(
tokens[_data.fromToken][_data.beneficiary] >= _data.fromAmount,
"Balance insufficient"
);
require(
_data.fromToken == BTCT_ADDR,
"fromToken != BTCT_ADDR"
);
tokens[_data.fromToken][_data.beneficiary] = tokens[_data.fromToken][
_data.beneficiary
].sub(_data.fromAmount);
_doSimpleSwap(_data);
}
function spFlow1Uniswap(
bool _fork,
address _factory,
bytes32 _initCode,
uint256 _amountIn,
uint256 _amountOutMin,
address[] calldata _path
) external nonReentrant returns (uint256 receivedAmount) {
address fromToken = _path[0];
address endToken = _path[_path.length - 1];
require(
tokens[fromToken][msg.sender] >= _amountIn,
"Balance insufficient"
);
require(fromToken == BTCT_ADDR, "fromToken != BTCT_ADDR");
require(endToken != ETHER, "Use path wBTC -> wETH");
uint256 preSwapBalance = IERC20(endToken).balanceOf(address(this));
tokens[fromToken][msg.sender] = tokens[fromToken][msg.sender].sub(
_amountIn
);
if (_fork) {
_doUniswapFork(
_factory,
_initCode,
_amountIn,
_amountOutMin,
_path
);
} else {
_doUniswap(_amountIn, _amountOutMin, _path);
}
receivedAmount = IERC20(endToken).balanceOf(address(this)).sub(
preSwapBalance
);
require(
receivedAmount >= _amountOutMin,
"Received < minimum"
);
tokens[endToken][msg.sender] = tokens[endToken][msg.sender].add(
receivedAmount
);
return receivedAmount;
}
function spFlow2Uniswap(
string calldata _destinationAddressForBTC,
bool _fork,
address _factory,
bytes32 _initCode,
uint256 _amountIn,
uint256 _amountOutMin,
address[] calldata _path
) external nonReentrant returns (uint256 receivedAmount) {
address fromToken = _path[0];
address endToken = _path[_path.length - 1];
require(
tokens[fromToken][msg.sender] >= _amountIn,
"Balance insufficient"
);
require(fromToken != ETHER, "Use path wETH -> wBTC");
require(endToken == BTCT_ADDR, "swap => BTCT");
uint256 preSwapBalance = IERC20(endToken).balanceOf(address(this));
tokens[fromToken][msg.sender] = tokens[fromToken][msg.sender].sub(
_amountIn
);
if (_fork) {
_doUniswapFork(
_factory,
_initCode,
_amountIn,
_amountOutMin,
_path
);
} else {
_doUniswap(_amountIn, _amountOutMin, _path);
}
receivedAmount = IERC20(endToken).balanceOf(address(this)).sub(
preSwapBalance
);
require(
receivedAmount >= _amountOutMin,
"Received < minimum"
);
require(
receivedAmount >= ip.minimumSwapAmountForWBTC(),
"receivedAmount < minimumSwapAmountForWBTC"
);
_spRecordPendingTx(_destinationAddressForBTC, receivedAmount);
return receivedAmount;
}
function spFlow2SimpleSwap(
string calldata _destinationAddressForBTC,
Utils.SimpleData calldata _data
) external nonReentrant returns (uint256 receivedAmount) {
require(_data.fromToken != BTCT_ADDR, "Must not swap from BTC token");
require(_data.toToken == BTCT_ADDR, "Must swap to BTC token");
require(
_data.beneficiary == address(this),
"beneficiary != swap contract"
);
require(
tokens[_data.fromToken][msg.sender] >= _data.fromAmount,
"Balance insufficient"
);
uint256 preSwapBalance = IERC20(_data.toToken).balanceOf(address(this));
tokens[_data.fromToken][msg.sender] = tokens[_data.fromToken][
msg.sender
].sub(_data.fromAmount);
_doSimpleSwap(_data);
receivedAmount = IERC20(_data.toToken).balanceOf(address(this)).sub(
preSwapBalance
);
require(
receivedAmount >= _data.expectedAmount,
"Received amount insufficient"
);
require(
receivedAmount >= ip.minimumSwapAmountForWBTC(),
"receivedAmount < minimumSwapAmountForWBTC"
);
_spRecordPendingTx(_destinationAddressForBTC, receivedAmount);
return receivedAmount;
}
function _doUniswapFork(
address _factory,
bytes32 _initCode,
uint256 _amountIn,
uint256 _amountOutMin,
address[] calldata _path
) internal {
IERC20(_path[0]).safeIncreaseAllowance(
IAugustusSwapper(paraswapAddress).getTokenTransferProxy(),
_amountIn
);
IParaswap(paraswapAddress).swapOnUniswapFork(
_factory,
_initCode,
_amountIn,
_amountOutMin,
_path
);
}
function _doUniswap(
uint256 _amountIn,
uint256 _amountOutMin,
address[] calldata _path
) internal {
IERC20(_path[0]).safeIncreaseAllowance(
IAugustusSwapper(paraswapAddress).getTokenTransferProxy(),
_amountIn
);
IParaswap(paraswapAddress).swapOnUniswap(
_amountIn,
_amountOutMin,
_path
);
}
function _doSimpleSwap(Utils.SimpleData calldata _data) internal {
IERC20(_data.fromToken).safeIncreaseAllowance(
IAugustusSwapper(paraswapAddress).getTokenTransferProxy(),
_data.fromAmount
);
IParaswap(paraswapAddress).simpleSwap(_data);
}
function _spRecordPendingTx(
string calldata _destinationAddressForBTC,
uint256 _btctAmount
) internal {
bytes32 ID = keccak256(
abi.encodePacked(
BTCT_ADDR,
swapCount,
_destinationAddressForBTC,
_btctAmount,
block.timestamp
)
);
spPendingTXs[swapCount] = spPendingTx(
ID,
_destinationAddressForBTC,
msg.sender,
_btctAmount,
block.timestamp
);
spCleanUpOldTXs();
swapCount = swapCount.add(1);
_reduceLimitBTCForSPFlow2(_btctAmount);
emit SwapTokensToBTC(
ID,
_destinationAddressForBTC,
msg.sender,
_btctAmount,
block.timestamp
);
}
function spCleanUpOldTXs() public {
uint256 max = oldestActiveIndex.add(ip.loopCount());
if (max >= swapCount) {
max = swapCount;
}
uint256 current = block.timestamp;
for (uint256 i = oldestActiveIndex; i < max; i++) {
if (spPendingTXs[i].Timestamp.add(ip.expirationTime()) < current) {
delete spPendingTXs[i];
oldestActiveIndex = i.add(1);
}
}
}
function spDeposit(address _token, uint256 _amount)
external
payable
nonReentrant
{
if (msg.value == 0) {
require(_token != ETHER);
require(_token != BTCT_ADDR);
uint256 initBalance = IERC20(_token).balanceOf(address(this));
IERC20(_token).safeTransferFrom(msg.sender, address(this), _amount);
uint256 received = IERC20(_token).balanceOf(address(this)).sub(initBalance);
tokens[_token][msg.sender] = tokens[_token][msg.sender].add(
received
);
emit Deposit(
_token,
msg.sender,
received,
tokens[_token][msg.sender],
block.timestamp
);
} else {
require(msg.value > 0);
IWETH(wETH).deposit{value: msg.value}();
tokens[wETH][msg.sender] = tokens[wETH][msg.sender].add(msg.value);
emit Deposit(
ETHER,
msg.sender,
msg.value,
tokens[wETH][msg.sender],
block.timestamp
);
}
}
function redeemEther(uint256 _amount) external nonReentrant {
require(tokens[wETH][msg.sender] >= _amount);
IWETH(wETH).withdraw(_amount);
tokens[wETH][msg.sender] = tokens[wETH][msg.sender].sub(_amount);
address payable sender = payable(msg.sender);
(
bool success,
) = sender.call{value: _amount}("");
require(success, "receiver rejected ETH transfer");
emit Withdraw(
ETHER,
msg.sender,
_amount,
tokens[wETH][msg.sender],
block.timestamp
);
}
receive() external payable {
assert(msg.sender == wETH);
}
function redeemERC20Token(address _token, uint256 _amount)
external
nonReentrant
{
require(tokens[_token][msg.sender] >= _amount, "Insufficient Balance");
tokens[_token][msg.sender] = tokens[_token][msg.sender].sub(_amount);
_safeTransfer(_token, msg.sender, _amount);
emit Withdraw(
_token,
msg.sender,
_amount,
tokens[_token][msg.sender],
block.timestamp
);
}
function recordUTXOSweepMinerFee(uint256 _minerFee, bytes32 _txid)
public
override
onlyOwner
returns (bool)
{
require(!isTxUsed(_txid), "The txid is already used");
floatAmountOf[address(0)] = floatAmountOf[address(0)].sub(
_minerFee,
"12"
);
_addUsedTx(_txid);
return true;
}
function churn(
address _newOwner,
address[] memory _nodes,
bool[] memory _isRemoved,
uint8 _churnedInCount,
uint8 _tssThreshold
) external override onlyOwner returns (bool) {
require(
_tssThreshold >= tssThreshold && _tssThreshold <= 2**8 - 1,
"01"
);
require(
_churnedInCount >= _tssThreshold + uint8(1),
"02"
);
require(
_nodes.length == _isRemoved.length,
"05"
);
transferOwnership(_newOwner);
for (uint256 i = 0; i < _nodes.length; i++) {
if (!_isRemoved[i]) {
if (nodes[_nodes[i]] == uint8(0)) {
nodeAddrs.push(_nodes[i]);
}
if (nodes[_nodes[i]] != uint8(1)) {
activeNodeCount++;
}
nodes[_nodes[i]] = uint8(1);
} else {
activeNodeCount--;
nodes[_nodes[i]] = uint8(2);
}
}
require(activeNodeCount <= 100, "Stored node size should be <= 100");
churnedInCount = _churnedInCount;
tssThreshold = _tssThreshold;
return true;
}
function balanceOf(address _token, address _user)
public
view
returns (uint256)
{
return tokens[_token][_user];
}
function spGetPendingSwaps()
external
view
returns (spPendingTx[] memory data)
{
uint256 index = 0;
data = new spPendingTx[](swapCount.sub(oldestActiveIndex));
for (uint256 i = oldestActiveIndex.add(1); i <= swapCount; i++) {
data[index] = spPendingTXs[index.add(oldestActiveIndex)];
index = index.add(1);
}
return data;
}
function isTxUsed(bytes32 _txid) public view override returns (bool) {
return used[_txid];
}
function getCurrentPriceLP()
public
view
override
returns (uint256 nowPrice)
{
(uint256 reserveA, uint256 reserveB) = getFloatReserve(
address(0),
BTCT_ADDR
);
uint256 totalLPs = lpToken.totalSupply();
nowPrice = totalLPs == 0
? lpDivisor
: (reserveA.add(reserveB)).mul(lpDivisor).div(totalLPs);
return nowPrice;
}
function getFloatReserve(address _tokenA, address _tokenB)
public
view
override
returns (uint256 reserveA, uint256 reserveB)
{
(reserveA, reserveB) = (floatAmountOf[_tokenA], floatAmountOf[_tokenB]);
}
function getActiveNodes() public view override returns (address[] memory) {
uint256 count = 0;
address[] memory _nodes = new address[](activeNodeCount);
for (uint256 i = 0; i < nodeAddrs.length; i++) {
if (nodes[nodeAddrs[i]] == uint8(1)) {
_nodes[count] = nodeAddrs[i];
count++;
}
}
return _nodes;
}
function isNodeStake(address _user) public view override returns (bool) {
if (nodes[_user] == uint8(1)) {
return true;
}
return false;
}
function _issueLPTokensForFloat(
address _token,
bytes32 _transaction,
bytes32 _txid
) internal returns (bool) {
require(!isTxUsed(_txid), "06");
require(_transaction != 0x0, "07");
(address to, uint256 amountOfFloat) = _splitToValues(_transaction);
uint256 nowPrice = getCurrentPriceLP();
uint256 amountOfLP = amountOfFloat.mul(lpDivisor).div(nowPrice);
lpToken.mint(to, amountOfLP);
_addFloat(_token, amountOfFloat);
_addUsedTx(_txid);
emit IssueLPTokensForFloat(
to,
amountOfFloat,
amountOfLP,
nowPrice,
ip.depositFeesBPS(),
_txid
);
return true;
}
function _burnLPTokensForFloat(
address _token,
bytes32 _transaction,
uint256 _minerFee,
bytes32 _txid
) internal returns (bool) {
require(!isTxUsed(_txid), "06");
require(_transaction != 0x0, "07");
(address to, uint256 amountOfLP) = _splitToValues(_transaction);
uint256 nowPrice = getCurrentPriceLP();
uint256 amountOfFloat = amountOfLP.mul(nowPrice).div(lpDivisor);
uint256 withdrawalFees = amountOfFloat.mul(ip.withdrawalFeeBPS()).div(
10000
);
require(
amountOfFloat.sub(withdrawalFees) >= _minerFee,
"09"
);
uint256 withdrawal = amountOfFloat.sub(withdrawalFees).sub(_minerFee);
(uint256 reserveA, uint256 reserveB) = getFloatReserve(
address(0),
BTCT_ADDR
);
if (_token == address(0)) {
require(
reserveA >= amountOfFloat.sub(withdrawalFees),
"08"
);
} else if (_token == BTCT_ADDR) {
require(
reserveB >= amountOfFloat.sub(withdrawalFees),
"12"
);
}
_rewardsCollection(_token, withdrawalFees);
_removeFloat(_token, amountOfFloat);
_addUsedTx(_txid);
if (_token == BTCT_ADDR) {
_safeTransfer(_token, to, withdrawal);
}
require(lpToken.burn(amountOfLP));
emit BurnLPTokensForFloat(
to,
amountOfLP,
amountOfFloat,
nowPrice,
withdrawal,
_txid
);
return true;
}
function _addFloat(address _token, uint256 _amount) internal {
floatAmountOf[_token] = floatAmountOf[_token].add(_amount);
}
function _removeFloat(address _token, uint256 _amount) internal {
floatAmountOf[_token] = floatAmountOf[_token].sub(
_amount,
"10"
);
}
function _swap(
address _sourceToken,
address _destToken,
uint256 _swapAmount
) internal {
floatAmountOf[_destToken] = floatAmountOf[_destToken].sub(
_swapAmount,
"11"
);
floatAmountOf[_sourceToken] = floatAmountOf[_sourceToken].add(
_swapAmount
);
emit Swap(_sourceToken, _destToken, _swapAmount);
}
function _safeTransfer(
address _token,
address _to,
uint256 _amount
) internal {
if (_token == BTCT_ADDR) {
_amount = _amount.mul(convertScale);
}
IERC20(_token).safeTransfer(_to, _amount);
}
function _rewardsCollection(address _feesToken, uint256 _rewardsAmount)
internal
{
if (_rewardsAmount == 0) return;
uint256 nowPrice = getCurrentPriceLP();
floatAmountOf[_feesToken] = floatAmountOf[_feesToken].add(
_rewardsAmount
);
uint256 amountForNodes = _rewardsAmount.mul(ip.nodeRewardsRatio()).div(
100
);
uint256 amountLPTokensForNode = amountForNodes.mul(lpDivisor).div(
nowPrice
);
lpToken.mint(sbBTCPool, amountLPTokensForNode);
emit RewardsCollection(
_feesToken,
_rewardsAmount,
amountLPTokensForNode,
nowPrice
);
}
function _addUsedTx(bytes32 _txid) internal {
used[_txid] = true;
}
function _updateLimitBTCForSPFlow2() internal {
limitBTCForSPFlow2 = floatAmountOf[address(0)];
}
function _reduceLimitBTCForSPFlow2(uint256 _amount) internal {
if (limitBTCForSPFlow2 == 0) {
_updateLimitBTCForSPFlow2();
}
limitBTCForSPFlow2 = limitBTCForSPFlow2.sub(
_amount,
"12"
);
}
function _addUsedTxs(bytes32[] memory _txids) internal {
for (uint256 i = 0; i < _txids.length; i++) {
used[_txids[i]] = true;
}
}
function _splitToValues(bytes32 _data)
internal
pure
returns (address, uint256)
{
return (
address(uint160(uint256(_data))),
uint256(uint96(bytes12(_data)))
);
}
}
文件 17 的 17:Utils.sol
pragma solidity >=0.6.0 <=0.8.9;
library Utils {
struct SellData {
address fromToken;
uint256 fromAmount;
uint256 toAmount;
uint256 expectedAmount;
address payable beneficiary;
Utils.Path[] path;
address payable partner;
uint256 feePercent;
bytes permit;
uint256 deadline;
bytes16 uuid;
}
struct MegaSwapSellData {
address fromToken;
uint256 fromAmount;
uint256 toAmount;
uint256 expectedAmount;
address payable beneficiary;
Utils.MegaSwapPath[] path;
address payable partner;
uint256 feePercent;
bytes permit;
uint256 deadline;
bytes16 uuid;
}
struct SimpleData {
address fromToken;
address toToken;
uint256 fromAmount;
uint256 toAmount;
uint256 expectedAmount;
address[] callees;
bytes exchangeData;
uint256[] startIndexes;
uint256[] values;
address payable beneficiary;
address payable partner;
uint256 feePercent;
bytes permit;
uint256 deadline;
bytes16 uuid;
}
struct Adapter {
address payable adapter;
uint256 percent;
uint256 networkFee;
Route[] route;
}
struct Route {
uint256 index;
address targetExchange;
uint percent;
bytes payload;
uint256 networkFee;
}
struct MegaSwapPath {
uint256 fromAmountPercent;
Path[] path;
}
struct Path {
address to;
uint256 totalNetworkFee;
Adapter[] adapters;
}
}
{
"compilationTarget": {
"contracts/SwapContract.sol": "SwapContract"
},
"evmVersion": "london",
"libraries": {},
"metadata": {
"bytecodeHash": "ipfs"
},
"optimizer": {
"enabled": true,
"runs": 200
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"_lpToken","type":"address"},{"internalType":"address","name":"_btct","type":"address"},{"internalType":"address","name":"_wETH","type":"address"},{"internalType":"address","name":"_sbBTCPool","type":"address"},{"internalType":"address","name":"_params","type":"address"},{"internalType":"address","name":"_swapRewards","type":"address"},{"internalType":"uint256","name":"_existingBTCFloat","type":"uint256"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amountOfLP","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amountOfFloat","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"currentPriceLP","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"withdrawal","type":"uint256"},{"indexed":false,"internalType":"bytes32","name":"txid","type":"bytes32"}],"name":"BurnLPTokensForFloat","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"balance","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"Timestamp","type":"uint256"}],"name":"Deposit","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"rewardLPTsForNodes","type":"uint256"}],"name":"DistributeNodeRewards","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"amountOfFloat","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amountOfLP","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"currentPriceLP","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"depositFees","type":"uint256"},{"indexed":false,"internalType":"bytes32","name":"txid","type":"bytes32"}],"name":"IssueLPTokensForFloat","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"feesToken","type":"address"},{"indexed":false,"internalType":"uint256","name":"rewards","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amountLPTokensForNode","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"currentPriceLP","type":"uint256"}],"name":"RewardsCollection","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"from","type":"address"},{"indexed":false,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Swap","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bytes32","name":"SwapID","type":"bytes32"},{"indexed":false,"internalType":"string","name":"DestAddr","type":"string"},{"indexed":false,"internalType":"address","name":"RefundAddr","type":"address"},{"indexed":false,"internalType":"uint256","name":"AmountWBTC","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"Timestamp","type":"uint256"}],"name":"SwapTokensToBTC","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"balance","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"Timestamp","type":"uint256"}],"name":"Withdraw","type":"event"},{"inputs":[],"name":"BTCT_ADDR","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"activeNodeCount","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"address","name":"_user","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_newOwner","type":"address"},{"internalType":"address[]","name":"_nodes","type":"address[]"},{"internalType":"bool[]","name":"_isRemoved","type":"bool[]"},{"internalType":"uint8","name":"_churnedInCount","type":"uint8"},{"internalType":"uint8","name":"_tssThreshold","type":"uint8"}],"name":"churn","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"churnedInCount","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_incomingAmount","type":"uint256"},{"internalType":"uint256","name":"_minerFee","type":"uint256"},{"internalType":"uint256","name":"_rewardsAmount","type":"uint256"},{"internalType":"address[]","name":"_spenders","type":"address[]"},{"internalType":"uint256[]","name":"_swapAmounts","type":"uint256[]"},{"internalType":"bool","name":"_isUpdatelimitBTCForSPFlow2","type":"bool"}],"name":"collectSwapFeesForBTC","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"getActiveNodes","outputs":[{"internalType":"address[]","name":"","type":"address[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getCurrentPriceLP","outputs":[{"internalType":"uint256","name":"nowPrice","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_tokenA","type":"address"},{"internalType":"address","name":"_tokenB","type":"address"}],"name":"getFloatReserve","outputs":[{"internalType":"uint256","name":"reserveA","type":"uint256"},{"internalType":"uint256","name":"reserveB","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ip","outputs":[{"internalType":"contract IParams","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_user","type":"address"}],"name":"isNodeStake","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"_txid","type":"bytes32"}],"name":"isTxUsed","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"limitBTCForSPFlow2","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lpToken","outputs":[{"internalType":"contract IBurnableToken","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32[]","name":"_addressesAndAmounts","type":"bytes32[]"},{"internalType":"uint256","name":"_totalSwapped","type":"uint256"},{"internalType":"uint256","name":"_rewardsAmount","type":"uint256"},{"internalType":"bytes32[]","name":"_usedTxIds","type":"bytes32[]"}],"name":"multiRecordSkyPoolsTX","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_destToken","type":"address"},{"internalType":"bytes32[]","name":"_addressesAndAmounts","type":"bytes32[]"},{"internalType":"uint256","name":"_totalSwapped","type":"uint256"},{"internalType":"uint256","name":"_rewardsAmount","type":"uint256"},{"internalType":"bytes32[]","name":"_redeemedFloatTxIds","type":"bytes32[]"}],"name":"multiTransferERC20TightlyPacked","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"oldestActiveIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"paraswapAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"bytes32","name":"_addressesAndAmountOfFloat","type":"bytes32"},{"internalType":"bytes32","name":"_txid","type":"bytes32"}],"name":"recordIncomingFloat","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"bytes32","name":"_addressesAndAmountOfLPtoken","type":"bytes32"},{"internalType":"uint256","name":"_minerFee","type":"uint256"},{"internalType":"bytes32","name":"_txid","type":"bytes32"}],"name":"recordOutcomingFloat","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_totalSwapped","type":"uint256"},{"internalType":"uint256","name":"_rewardsAmount","type":"uint256"},{"internalType":"bytes32[]","name":"_usedTxIds","type":"bytes32[]"}],"name":"recordSkyPoolsTX","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_minerFee","type":"uint256"},{"internalType":"bytes32","name":"_txid","type":"bytes32"}],"name":"recordUTXOSweepMinerFee","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"redeemERC20Token","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"redeemEther","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"sbBTCPool","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_destToken","type":"address"},{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"uint256","name":"_totalSwapped","type":"uint256"},{"internalType":"uint256","name":"_rewardsAmount","type":"uint256"},{"internalType":"bytes32[]","name":"_redeemedFloatTxIds","type":"bytes32[]"}],"name":"singleTransferERC20","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"spCleanUpOldTXs","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"spDeposit","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"fromToken","type":"address"},{"internalType":"address","name":"toToken","type":"address"},{"internalType":"uint256","name":"fromAmount","type":"uint256"},{"internalType":"uint256","name":"toAmount","type":"uint256"},{"internalType":"uint256","name":"expectedAmount","type":"uint256"},{"internalType":"address[]","name":"callees","type":"address[]"},{"internalType":"bytes","name":"exchangeData","type":"bytes"},{"internalType":"uint256[]","name":"startIndexes","type":"uint256[]"},{"internalType":"uint256[]","name":"values","type":"uint256[]"},{"internalType":"address payable","name":"beneficiary","type":"address"},{"internalType":"address payable","name":"partner","type":"address"},{"internalType":"uint256","name":"feePercent","type":"uint256"},{"internalType":"bytes","name":"permit","type":"bytes"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"bytes16","name":"uuid","type":"bytes16"}],"internalType":"struct Utils.SimpleData","name":"_data","type":"tuple"}],"name":"spFlow1SimpleSwap","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_fork","type":"bool"},{"internalType":"address","name":"_factory","type":"address"},{"internalType":"bytes32","name":"_initCode","type":"bytes32"},{"internalType":"uint256","name":"_amountIn","type":"uint256"},{"internalType":"uint256","name":"_amountOutMin","type":"uint256"},{"internalType":"address[]","name":"_path","type":"address[]"}],"name":"spFlow1Uniswap","outputs":[{"internalType":"uint256","name":"receivedAmount","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_destinationAddressForBTC","type":"string"},{"components":[{"internalType":"address","name":"fromToken","type":"address"},{"internalType":"address","name":"toToken","type":"address"},{"internalType":"uint256","name":"fromAmount","type":"uint256"},{"internalType":"uint256","name":"toAmount","type":"uint256"},{"internalType":"uint256","name":"expectedAmount","type":"uint256"},{"internalType":"address[]","name":"callees","type":"address[]"},{"internalType":"bytes","name":"exchangeData","type":"bytes"},{"internalType":"uint256[]","name":"startIndexes","type":"uint256[]"},{"internalType":"uint256[]","name":"values","type":"uint256[]"},{"internalType":"address payable","name":"beneficiary","type":"address"},{"internalType":"address payable","name":"partner","type":"address"},{"internalType":"uint256","name":"feePercent","type":"uint256"},{"internalType":"bytes","name":"permit","type":"bytes"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"bytes16","name":"uuid","type":"bytes16"}],"internalType":"struct Utils.SimpleData","name":"_data","type":"tuple"}],"name":"spFlow2SimpleSwap","outputs":[{"internalType":"uint256","name":"receivedAmount","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_destinationAddressForBTC","type":"string"},{"internalType":"bool","name":"_fork","type":"bool"},{"internalType":"address","name":"_factory","type":"address"},{"internalType":"bytes32","name":"_initCode","type":"bytes32"},{"internalType":"uint256","name":"_amountIn","type":"uint256"},{"internalType":"uint256","name":"_amountOutMin","type":"uint256"},{"internalType":"address[]","name":"_path","type":"address[]"}],"name":"spFlow2Uniswap","outputs":[{"internalType":"uint256","name":"receivedAmount","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"spGetPendingSwaps","outputs":[{"components":[{"internalType":"bytes32","name":"SwapID","type":"bytes32"},{"internalType":"string","name":"DestAddr","type":"string"},{"internalType":"address","name":"RefundAddr","type":"address"},{"internalType":"uint256","name":"AmountWBTC","type":"uint256"},{"internalType":"uint256","name":"Timestamp","type":"uint256"}],"internalType":"struct SwapContract.spPendingTx[]","name":"data","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"spPendingTXs","outputs":[{"internalType":"bytes32","name":"SwapID","type":"bytes32"},{"internalType":"string","name":"DestAddr","type":"string"},{"internalType":"address","name":"RefundAddr","type":"address"},{"internalType":"uint256","name":"AmountWBTC","type":"uint256"},{"internalType":"uint256","name":"Timestamp","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"sw","outputs":[{"internalType":"contract ISwapRewards","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"swapCount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"}],"name":"tokens","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"tssThreshold","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"wETH","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"whitelist","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"stateMutability":"payable","type":"receive"}]