InstaContract/contracts/protocols/InstaKyber.sol

173 lines
4.3 KiB
Solidity
Raw Normal View History

pragma solidity ^0.4.24;
2018-11-25 21:39:47 +00:00
library SafeMath {
2018-11-25 21:39:47 +00:00
function mul(uint256 a, uint256 b) internal pure returns (uint256) {
if (a == 0) {
return 0;
}
uint256 c = a * b;
require(c / a == b, "Assertion Failed");
return c;
}
function div(uint256 a, uint256 b) internal pure returns (uint256) {
require(b > 0, "Assertion Failed");
uint256 c = a / b;
return c;
}
}
interface IERC20 {
function balanceOf(address who) external view returns (uint256);
function transfer(address to, uint256 value) external returns (bool);
function approve(address spender, uint256 value) external returns (bool);
function transferFrom(address from, address to, uint256 value) external returns (bool);
}
interface AddressRegistry {
function getAddr(string name) external view returns(address);
}
interface Kyber {
function trade(
address src,
uint srcAmount,
address dest,
address destAddress,
uint maxDestAmount,
uint minConversionRate,
address walletId
) external payable returns (uint);
2018-10-29 11:19:01 +00:00
function getExpectedRate(
address src,
address dest,
uint srcQty
) external view returns (uint, uint);
}
contract Registry {
address public addressRegistry;
modifier onlyAdmin() {
require(
msg.sender == getAddress("admin"),
"Permission Denied"
);
_;
}
2018-10-29 11:19:01 +00:00
function getAddress(string name) internal view returns(address) {
AddressRegistry addrReg = AddressRegistry(addressRegistry);
return addrReg.getAddr(name);
}
}
contract Trade is Registry {
using SafeMath for uint;
using SafeMath for uint256;
2018-10-29 11:19:01 +00:00
event KyberTrade(
address src,
uint srcAmt,
address dest,
uint destAmt,
address beneficiary,
2018-10-29 11:19:01 +00:00
uint minConversionRate,
address affiliate
);
2018-11-25 19:26:55 +00:00
function getExpectedPrice(
address src,
address dest,
uint srcAmt
) public view returns (uint, uint)
{
Kyber kyberFunctions = Kyber(getAddress("kyber"));
return kyberFunctions.getExpectedRate(
src,
dest,
srcAmt
);
}
function approveKyber(address[] tokenArr) public {
for (uint i = 0; i < tokenArr.length; i++) {
IERC20 tokenFunctions = IERC20(tokenArr[i]);
tokenFunctions.approve(getAddress("kyber"), 2**256 - 1);
}
}
function executeTrade(
address src, // token to sell
address dest, // token to buy
uint srcAmt, // amount of token for sell
uint minConversionRate, // minimum slippage rate
uint maxDestAmt // max amount of dest token
) public payable returns (uint destAmt)
{
2018-11-25 19:26:55 +00:00
uint ethQty = getToken(msg.sender, src, srcAmt);
// Interacting with Kyber Proxy Contract
2018-10-29 11:19:01 +00:00
Kyber kyberFunctions = Kyber(getAddress("kyber"));
destAmt = kyberFunctions.trade.value(ethQty)(
src,
2018-11-25 19:26:55 +00:00
srcAmt,
dest,
2018-11-25 19:26:55 +00:00
msg.sender,
maxDestAmt,
2018-10-29 11:19:01 +00:00
minConversionRate,
getAddress("admin")
);
2018-11-25 21:39:47 +00:00
// maxDestAmt usecase implementated
if (src == getAddress("eth") && address(this).balance > 0) {
msg.sender.transfer(address(this).balance);
} else {
IERC20 srcTkn = IERC20(src);
uint srcBal = srcTkn.balanceOf(address(this));
if (srcBal > 0) {
srcTkn.transfer(msg.sender, srcBal);
}
}
emit KyberTrade(
src,
2018-11-25 19:26:55 +00:00
srcAmt,
dest,
destAmt,
2018-11-25 19:26:55 +00:00
msg.sender,
2018-10-29 11:19:01 +00:00
minConversionRate,
getAddress("admin")
);
}
2018-11-25 19:26:55 +00:00
function getToken(address trader, address src, uint srcAmt) internal returns (uint ethQty) {
if (src == getAddress("eth")) {
require(msg.value == srcAmt, "Invalid Operation");
ethQty = srcAmt;
} else {
IERC20 tokenFunctions = IERC20(src);
tokenFunctions.transferFrom(trader, address(this), srcAmt);
ethQty = 0;
}
}
}
2018-11-24 16:34:08 +00:00
contract InstaKyber is Trade {
constructor(address rAddr) public {
addressRegistry = rAddr;
}
2018-11-25 21:39:47 +00:00
function () public payable {}
}