Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Smart token #9

Open
wants to merge 53 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
53 commits
Select commit Hold shift + click to select a range
82682a2
Almost done with OneSplitSmartTokenView
krboktv Mar 30, 2020
66e7061
Merge pull request #1 from CryptoManiacsZone/master
krboktv Mar 30, 2020
08cba0b
Merge pull request #2 from krboktv/master
krboktv Mar 30, 2020
79074c3
fix
krboktv Mar 31, 2020
bea92a7
done with SmartTokenView contract
krboktv Mar 31, 2020
35cb84f
add _swapFromSmartToken function
krboktv Mar 31, 2020
3350e6e
add _swapToSmartToken function
krboktv Mar 31, 2020
dc2a84c
update full contract
krboktv Apr 1, 2020
323ef6b
fix msg.sender -> address(this)
krboktv Apr 1, 2020
536b05d
fix minReturnAmount
krboktv Apr 1, 2020
1488825
complete some <todo>
krboktv Apr 1, 2020
f728277
edit distribution
krboktv Apr 2, 2020
98a6368
reservers -> connectors + remove experimental encoder
krboktv Apr 2, 2020
dc408b5
get version -> staticcall
krboktv Apr 3, 2020
1bf38bd
fix leftover calculation
krboktv Apr 3, 2020
9b49195
safe fund swap
krboktv Apr 3, 2020
e79ff42
refactor
krboktv Apr 3, 2020
d113487
handle "lose funds" behavior
krboktv Apr 4, 2020
f5a24bd
some changes and safe calls
krboktv Apr 4, 2020
1cab6d3
disable flags -> 0
krboktv Apr 4, 2020
e3995a4
remove useless code
krboktv Apr 4, 2020
bf2d661
refactor
krboktv Apr 4, 2020
a975fb6
Some fixes
k06a Apr 4, 2020
04fc628
Safe Bancor swap added
k06a Apr 4, 2020
edec249
Fix Bancor safe swap
k06a Apr 4, 2020
c38acbe
fix shadowed naming
krboktv Apr 4, 2020
8f7038a
fix
krboktv Apr 4, 2020
f3cd14c
fixes
krboktv Apr 4, 2020
705a818
fix
krboktv Apr 5, 2020
25807f4
Fix bug
k06a Apr 5, 2020
014a5af
Revert abi and bin
k06a Apr 5, 2020
34d6e0a
Fix call
k06a Apr 5, 2020
959fc5f
Fix bug in view call
k06a Apr 5, 2020
d58fb9f
Fix
k06a Apr 5, 2020
978a652
Merge pull request #3 from CryptoManiacsZone/master
krboktv Apr 11, 2020
80743c2
unworkable uniswap pool token contract
krboktv Apr 11, 2020
e4849c9
modified formula and optimized
krboktv Apr 13, 2020
82e511b
add comment
krboktv Apr 13, 2020
c670567
done with view contract
krboktv Apr 21, 2020
ae2651a
almost done with swap
krboktv Apr 21, 2020
b50e5a1
almost done with curve susd pool token
krboktv Apr 26, 2020
0579ccd
rename vars
krboktv Apr 26, 2020
e888c9a
change snx curve address
krboktv Apr 27, 2020
125fc8c
some changes with weights
krboktv Apr 27, 2020
6bc1585
fix toToken split amount
krboktv Apr 27, 2020
61339c0
gas optimizations
krboktv Apr 27, 2020
8d0ed0f
fix curveSynthetix
krboktv Apr 27, 2020
af8a4df
optimize
krboktv Apr 27, 2020
f026063
Anton's refactoring
krboktv Apr 27, 2020
1dc5c72
fix
krboktv Apr 27, 2020
4cc1d3f
Merge pull request #4 from CryptoManiacsZone/master
krboktv May 2, 2020
36d126e
Merge branch 'master' into feature/curve-susd
krboktv May 2, 2020
8f54d0e
Merge branch 'feature/curve-susd' into smart-token
krboktv May 2, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
almost done with curve susd pool token
  • Loading branch information
krboktv committed Apr 26, 2020
commit b50e5a1df3d1e436b97d2b51a8d3afcdb84ee95e
1 change: 1 addition & 0 deletions contracts/IOneSplit.sol
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ contract IOneSplitConsts {
uint256 public constant FLAG_DISABLE_IDLE = 0x800000;
uint256 public constant FLAG_DISABLE_UNISWAP_POOL_TOKEN = 0x1000000;
uint256 public constant FLAG_DISABLE_BALANCER_POOL_TOKEN = 0x2000000;
uint256 public constant FLAG_DISABLE_CURVE_SUSD_POOL_TOKEN = 0x4000000;
}


Expand Down
314 changes: 314 additions & 0 deletions contracts/OneSplitCurveSusdPoolToken.sol
Original file line number Diff line number Diff line change
@@ -0,0 +1,314 @@
pragma solidity ^0.5.0;

import "./OneSplitBase.sol";
import "./interface/ICurve.sol";


contract OneSplitCurveSusdPoolTokenBase {
using SafeMath for uint256;
using UniversalERC20 for IERC20;

ISusdCurve curve = ISusdCurve(0xA5407eAE9Ba41422680e2e00537571bcC53efBfD);
IERC20 curveSusdToken = IERC20(0xC25a3A3b969415c80451098fa907EC722572917F);

struct TokenInfo {
IERC20 token;
uint256 reserveBalance;
}

struct PoolTokenDetails {
TokenInfo[] tokens;
uint256 totalSupply;
}

function _getPoolDetails()
internal
view
returns(PoolTokenDetails memory details)
{
details.tokens = new TokenInfo[](4);
details.totalSupply = curveSusdToken.totalSupply();
for (uint256 i = 0; i < 4; i++) {
details.tokens[i].token = IERC20(curve.coins(int128(i)));
details.tokens[i].reserveBalance = curve.balances(int128(i));
}
}

}

contract OneSplitCurveSusdPoolTokenView is OneSplitBaseView, OneSplitCurveSusdPoolTokenBase {

function getExpectedReturn(
IERC20 fromToken,
IERC20 toToken,
uint256 amount,
uint256 parts,
uint256 disableFlags
)
internal
returns (
uint256 returnAmount,
uint256[] memory distribution
)
{
if (fromToken == toToken) {
return (amount, new uint256[](DEXES_COUNT));
}


if (!disableFlags.check(FLAG_DISABLE_CURVE_SUSD_POOL_TOKEN)) {
if (fromToken == curveSusdToken) {
return _getExpectedReturnFromCurveSusdPoolToken(
fromToken,
toToken,
amount,
parts,
FLAG_DISABLE_CURVE_SUSD_POOL_TOKEN
);
}

if (toToken == curveSusdToken) {
return _getExpectedReturnToCurveSusdPoolToken(
fromToken,
toToken,
amount,
parts,
FLAG_DISABLE_CURVE_SUSD_POOL_TOKEN
);
}
}

return super.getExpectedReturn(
fromToken,
toToken,
amount,
parts,
disableFlags
);
}

function _getExpectedReturnFromCurveSusdPoolToken(
IERC20, // poolToken
IERC20 toToken,
uint256 amount,
uint256 parts,
uint256 disableFlags
)
private
returns (
uint256 returnAmount,
uint256[] memory distribution
)
{
distribution = new uint256[](DEXES_COUNT);

PoolTokenDetails memory details = _getPoolDetails();

uint256 ratio = amount.mul(1e18).div(details.totalSupply);
for (uint i = 0; i < 4; i++) {
uint256 tokenAmountOut = details.tokens[i].reserveBalance.mul(ratio).div(1e18);

if (details.tokens[i].token == toToken) {
returnAmount = returnAmount.add(tokenAmountOut);
continue;
}

(uint256 ret, uint256[] memory dist) = getExpectedReturn(
details.tokens[i].token,
toToken,
tokenAmountOut,
parts,
disableFlags
);

returnAmount = returnAmount.add(ret);

for (uint j = 0; j < distribution.length; j++) {
distribution[j] |= dist[j] << (i * 8);
}
}

return (returnAmount, distribution);
}

function _getExpectedReturnToCurveSusdPoolToken(
IERC20 fromToken,
IERC20, // poolToken
uint256 amount,
uint256 parts,
uint256 disableFlags
)
private
returns (
uint256 returnAmount,
uint256[] memory distribution
)
{
distribution = new uint256[](DEXES_COUNT);

PoolTokenDetails memory details = _getPoolDetails();

uint256[4] memory tokenAmounts;
uint256[] memory dist;
uint256 exchangeAmountPart = amount.div(4);
for (uint i = 0; i < 4; i++) {

if (details.tokens[i].token != fromToken) {
(tokenAmounts[i], dist) = getExpectedReturn(
fromToken,
details.tokens[i].token,
i != 3 ? exchangeAmountPart : exchangeAmountPart.add(exchangeAmountPart % 4),
parts,
disableFlags
);

for (uint j = 0; j < distribution.length; j++) {
distribution[j] |= dist[j] << (i * 8);
}
} else {
tokenAmounts[i] = i != 3 ? exchangeAmountPart : exchangeAmountPart.add(exchangeAmountPart % 4);
}
}

returnAmount = curve.calc_token_amount(tokenAmounts, true);

return (returnAmount, distribution);
}

}


contract OneSplitCurveSusdPoolToken is OneSplitBase, OneSplitCurveSusdPoolTokenBase {
function _swap(
IERC20 fromToken,
IERC20 toToken,
uint256 amount,
uint256[] memory distribution,
uint256 disableFlags
) internal {
if (fromToken == toToken) {
return;
}

if (!disableFlags.check(FLAG_DISABLE_CURVE_SUSD_POOL_TOKEN)) {
if (fromToken == curveSusdToken) {
return _swapFromCurveSusdPoolToken(
fromToken,
toToken,
amount,
distribution,
FLAG_DISABLE_CURVE_SUSD_POOL_TOKEN
);
}

if (toToken == curveSusdToken) {
return _swapToCurveSusdPoolToken(
fromToken,
toToken,
amount,
distribution,
FLAG_DISABLE_CURVE_SUSD_POOL_TOKEN
);
}
}

return super._swap(
fromToken,
toToken,
amount,
distribution,
disableFlags
);
}

function _swapFromCurveSusdPoolToken(
IERC20, // poolToken
IERC20 toToken,
uint256 amount,
uint256[] memory distribution,
uint256 disableFlags
) private {

PoolTokenDetails memory details = _getPoolDetails();

uint256 ratio = amount.mul(1e18).div(details.totalSupply);

uint256[4] memory minAmountsOut;
for (uint i = 0; i < 4; i++) {
minAmountsOut[i] = details.tokens[i].reserveBalance.mul(ratio).div(1e18).mul(995).div(1000); // 0.5% slippage;
}

curve.remove_liquidity(amount, minAmountsOut);

uint256[] memory dist = new uint256[](distribution.length);
for (uint i = 0; i < 4; i++) {

if (details.tokens[i].token == toToken) {
continue;
}

for (uint j = 0; j < distribution.length; j++) {
dist[j] = (distribution[j] >> (i * 8)) & 0xFF;
}

uint256 exchangeTokenAmount = details.tokens[i].token.balanceOf(address(this));

this.swap(
details.tokens[i].token,
toToken,
exchangeTokenAmount,
0,
dist,
disableFlags
);
}

}

function _swapToCurveSusdPoolToken(
IERC20 fromToken,
IERC20, // poolToken,
uint256 amount,
uint256[] memory distribution,
uint256 disableFlags
) private {
uint256[] memory dist = new uint256[](distribution.length);

PoolTokenDetails memory details = _getPoolDetails();

uint256[4] memory tokenAmounts;
uint256 exchangeAmountPart = amount.div(4);
for (uint i = 0; i < 4; i++) {

if (details.tokens[i].token != fromToken) {
uint256 tokenBalanceBefore = details.tokens[i].token.balanceOf(address(this));

for (uint j = 0; j < distribution.length; j++) {
dist[j] = (distribution[j] >> (i * 8)) & 0xFF;
}

this.swap(
fromToken,
details.tokens[i].token,
i != 3 ? exchangeAmountPart : exchangeAmountPart.add(exchangeAmountPart % 4),
0,
dist,
disableFlags
);

uint256 tokenBalanceAfter = details.tokens[i].token.balanceOf(address(this));

tokenAmounts[i] = tokenBalanceAfter.sub(tokenBalanceBefore);
} else {
tokenAmounts[i] = i != 3 ? exchangeAmountPart : exchangeAmountPart.add(exchangeAmountPart % 4);
}

_infiniteApproveIfNeeded(details.tokens[i].token, address(curve));
}

uint256 minAmount = curve.calc_token_amount(tokenAmounts, true);

// 0.5% slippage
curve.add_liquidity(tokenAmounts, minAmount.mul(995).div(1000));
}
}
13 changes: 13 additions & 0 deletions contracts/interface/ICurve.sol
Original file line number Diff line number Diff line change
Expand Up @@ -8,3 +8,16 @@ interface ICurve {
// solium-disable-next-line mixedcase
function exchange_underlying(int128 i, int128 j, uint256 dx, uint256 minDy) external;
}

interface ISusdCurve {

function coins(int128 arg0) external view returns (address);

function balances(int128 arg0) external view returns (uint256);

function add_liquidity(uint256[4] calldata amounts, uint256 min_mint_amount) external;

function remove_liquidity(uint256 _amount, uint256[4] calldata min_amounts) external;

function calc_token_amount(uint256[4] calldata amounts, bool deposit) external view returns (uint256);
}