mirror of
https://github.com/embarklabs/subspace.git
synced 2025-02-23 03:48:17 +00:00
141 lines
5.4 KiB
Solidity
141 lines
5.4 KiB
Solidity
/* solium-disable security/no-block-members */
|
|
pragma solidity >=0.5.0 <0.6.0;
|
|
|
|
import "../common/Ownable.sol";
|
|
import "../token/ERC20Token.sol";
|
|
import "./KyberNetworkProxy.sol";
|
|
|
|
/**
|
|
* @title KyberFeeBurner
|
|
* @dev Contract that holds assets for the purpose of trading them to SNT and burning them
|
|
* @dev Assets come from the Escrow contract fees
|
|
*/
|
|
contract KyberFeeBurner is Ownable {
|
|
|
|
address public SNT;
|
|
address public burnAddress;
|
|
address public walletId;
|
|
KyberNetworkProxy public kyberNetworkProxy;
|
|
|
|
// In Kyber's contracts, this is the address for ETH
|
|
address constant internal ETH_TOKEN_ADDRESS = 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE;
|
|
|
|
/**
|
|
* @param _snt Address of the SNT contract
|
|
* @param _burnAddress Address where to burn the assets
|
|
* @param _kyberNetworkProxy License contract instance address for arbitrators
|
|
* @param _walletId Wallet address to send part of the fees to (used for the fee sharing program)
|
|
*/
|
|
constructor(address _snt, address _burnAddress, address _kyberNetworkProxy, address _walletId) public {
|
|
SNT = _snt;
|
|
burnAddress = _burnAddress;
|
|
kyberNetworkProxy = KyberNetworkProxy(_kyberNetworkProxy);
|
|
walletId = _walletId;
|
|
}
|
|
|
|
event SNTAddressChanged(address sender, address prevSNTAddress, address newSNTAddress);
|
|
|
|
/**
|
|
* @dev Changes the SNT contract address
|
|
* @param _snt New SNT contract address
|
|
*/
|
|
function setSNT(address _snt) external onlyOwner {
|
|
emit SNTAddressChanged(msg.sender, SNT, _snt);
|
|
SNT = _snt;
|
|
}
|
|
|
|
event BurnAddressChanged(address sender, address prevBurnAddress, address newBurnAddress);
|
|
|
|
/**
|
|
* @dev Changes the burn address
|
|
* @param _burnAddress New burn address
|
|
*/
|
|
function setBurnAddress(address _burnAddress) external onlyOwner {
|
|
emit BurnAddressChanged(msg.sender, burnAddress, _burnAddress);
|
|
burnAddress = _burnAddress;
|
|
}
|
|
|
|
event KyberNetworkProxyAddressChanged(address sender, address prevKyberAddress, address newKyberAddress);
|
|
|
|
/**
|
|
* @dev Changes the KyberNetworkProxy contract address
|
|
* @param _kyberNetworkProxy New KyberNetworkProxy address
|
|
*/
|
|
function setKyberNetworkProxyAddress(address _kyberNetworkProxy) external onlyOwner {
|
|
emit KyberNetworkProxyAddressChanged(msg.sender, address(kyberNetworkProxy), _kyberNetworkProxy);
|
|
kyberNetworkProxy = KyberNetworkProxy(_kyberNetworkProxy);
|
|
}
|
|
|
|
event WalletIdChanged(address sender, address prevWalletId, address newWalletId);
|
|
|
|
/**
|
|
* @dev Changes the walletId address (for the fee sharing program)
|
|
* @param _walletId New walletId address
|
|
*/
|
|
function setWalletId(address _walletId) external onlyOwner {
|
|
emit WalletIdChanged(msg.sender, walletId, _walletId);
|
|
walletId = _walletId;
|
|
}
|
|
|
|
event Swap(address sender, address srcToken, address destToken, uint srcAmount, uint destAmount);
|
|
|
|
/**
|
|
* @dev Swaps the selected asset to SNT and transfers it to the burn address
|
|
* @param _token Address of the asset to trade
|
|
*/
|
|
function swap(address _token) external {
|
|
uint balance = 0;
|
|
uint minConversionRate = 0;
|
|
uint destAmount = 0;
|
|
|
|
if (_token == address(0)) {
|
|
balance = address(this).balance;
|
|
(minConversionRate,) = kyberNetworkProxy.getExpectedRate(ETH_TOKEN_ADDRESS, SNT, balance);
|
|
destAmount = kyberNetworkProxy.trade.value(balance)(ETH_TOKEN_ADDRESS, balance, SNT, burnAddress, 0 - uint256(1), minConversionRate, walletId);
|
|
emit Swap(msg.sender, ETH_TOKEN_ADDRESS, SNT, balance, destAmount);
|
|
} else {
|
|
ERC20Token t = ERC20Token(_token);
|
|
balance = t.balanceOf(address(this));
|
|
if (_token == SNT) {
|
|
require(t.transfer(burnAddress, balance), "SNT transfer failure");
|
|
emit Swap(msg.sender, SNT, SNT, balance, balance);
|
|
return;
|
|
} else {
|
|
// Mitigate ERC20 Approve front-running attack, by initially setting allowance to 0
|
|
require(ERC20Token(_token).approve(address(kyberNetworkProxy), 0), "Failed to reset approval");
|
|
|
|
// Set the spender's token allowance to tokenQty
|
|
require(ERC20Token(_token).approve(address(kyberNetworkProxy), balance), "Failed to approve trade amount");
|
|
|
|
(minConversionRate,) = kyberNetworkProxy.getExpectedRate(_token, SNT, balance);
|
|
destAmount = kyberNetworkProxy.trade(_token, balance, SNT, burnAddress, 0 - uint256(1), minConversionRate, walletId);
|
|
|
|
emit Swap(msg.sender, _token, SNT, balance, destAmount);
|
|
}
|
|
}
|
|
}
|
|
|
|
event EscapeTriggered(address sender, address token, uint amount);
|
|
|
|
/**
|
|
* @dev Exits the selected asset to the owner
|
|
* @param _token Address of the asset to exit
|
|
*/
|
|
function escape(address _token) external onlyOwner {
|
|
if (_token == address(0)) {
|
|
uint ethBalance = address(this).balance;
|
|
address(uint160(owner())).transfer(ethBalance);
|
|
emit EscapeTriggered(msg.sender, _token, ethBalance);
|
|
} else {
|
|
ERC20Token t = ERC20Token(_token);
|
|
uint tokenBalance = t.balanceOf(address(this));
|
|
require(t.transfer(owner(), tokenBalance), "Token transfer error");
|
|
emit EscapeTriggered(msg.sender, _token, tokenBalance);
|
|
}
|
|
}
|
|
|
|
function() payable external {
|
|
}
|
|
|
|
}
|