* add balancer v2 * fetch balancer v2 pools from subgraph * feat: initial stab at a Balancer V2 Sampler WIP * feat: add sampling for buys and fix build issues WIP [untested] * fix: BalancerV2Sampler implementation issues, works on Kovan * chore: BalancerV2 sampling boilerplate * fix: update Balancer V2 mainnet address * fix: consolidate differences between the 2 working branches * fix: use mainnet Balancer V2 subgraph * fix: stack too deep by minimizing and inline Balancer V2 vault interface * fix: address review comments and clean up * fix: sampler vault interface and pools cache assuming a pool has swaps * address more review comments * fix: TS type issues and add a comment about deadline argument * fix: pools_cache_tests incorrect token addresses, prettier incompat * fix: make ERC20BridgeSampler support BalancerV2 non view sampler fns * fix: use a struct for passing encoded bridge data for Balancer V2 * chore: add changelog entries * fix: improve gas accuracy of gas schedule for Balancer V2 & Maker Psm * fix: don't exclude sources with stale caches & wait for cache refresh * rebase * `@0x/asset-swapper`: Fix stack too deep errors in sampler Co-authored-by: Kim Persson <kimpersson88@gmail.com> Co-authored-by: Lawrence Forman <me@merklejerk.com>
190 lines
6.6 KiB
Solidity
190 lines
6.6 KiB
Solidity
// SPDX-License-Identifier: Apache-2.0
|
|
/*
|
|
|
|
Copyright 2021 ZeroEx Intl.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
pragma solidity ^0.6;
|
|
pragma experimental ABIEncoderV2;
|
|
|
|
import "./SamplerUtils.sol";
|
|
|
|
/// @dev Minimal Balancer V2 Vault interface
|
|
/// for documentation refer to https://github.com/balancer-labs/balancer-core-v2/blob/master/contracts/vault/interfaces/IVault.sol
|
|
interface IBalancerV2Vault {
|
|
enum SwapKind { GIVEN_IN, GIVEN_OUT }
|
|
|
|
struct BatchSwapStep {
|
|
bytes32 poolId;
|
|
uint256 assetInIndex;
|
|
uint256 assetOutIndex;
|
|
uint256 amount;
|
|
bytes userData;
|
|
}
|
|
|
|
struct FundManagement {
|
|
address sender;
|
|
bool fromInternalBalance;
|
|
address payable recipient;
|
|
bool toInternalBalance;
|
|
}
|
|
|
|
function queryBatchSwap(
|
|
SwapKind kind,
|
|
BatchSwapStep[] calldata swaps,
|
|
IAsset[] calldata assets,
|
|
FundManagement calldata funds
|
|
) external returns (int256[] memory assetDeltas);
|
|
}
|
|
interface IAsset {
|
|
// solhint-disable-previous-line no-empty-blocks
|
|
}
|
|
|
|
contract BalancerV2Sampler is SamplerUtils {
|
|
|
|
struct BalancerV2PoolInfo {
|
|
bytes32 poolId;
|
|
address vault;
|
|
}
|
|
|
|
/// @dev Sample sell quotes from Balancer V2.
|
|
/// @param poolInfo Struct with pool related data
|
|
/// @param takerToken Address of the taker token (what to sell).
|
|
/// @param makerToken Address of the maker token (what to buy).
|
|
/// @param takerTokenAmounts Taker token sell amount for each sample.
|
|
/// @return makerTokenAmounts Maker amounts bought at each taker token
|
|
/// amount.
|
|
function sampleSellsFromBalancerV2(
|
|
BalancerV2PoolInfo memory poolInfo,
|
|
address takerToken,
|
|
address makerToken,
|
|
uint256[] memory takerTokenAmounts
|
|
)
|
|
public
|
|
returns (uint256[] memory makerTokenAmounts)
|
|
{
|
|
_assertValidPair(makerToken, takerToken);
|
|
IBalancerV2Vault vault = IBalancerV2Vault(poolInfo.vault);
|
|
IAsset[] memory swapAssets = new IAsset[](2);
|
|
swapAssets[0] = IAsset(takerToken);
|
|
swapAssets[1] = IAsset(makerToken);
|
|
|
|
uint256 numSamples = takerTokenAmounts.length;
|
|
makerTokenAmounts = new uint256[](numSamples);
|
|
IBalancerV2Vault.FundManagement memory swapFunds =
|
|
_createSwapFunds();
|
|
|
|
for (uint256 i = 0; i < numSamples; i++) {
|
|
IBalancerV2Vault.BatchSwapStep[] memory swapSteps =
|
|
_createSwapSteps(poolInfo, takerTokenAmounts[i]);
|
|
|
|
try
|
|
// For sells we specify the takerToken which is what the vault will receive from the trade
|
|
vault.queryBatchSwap(IBalancerV2Vault.SwapKind.GIVEN_IN, swapSteps, swapAssets, swapFunds)
|
|
// amounts represent pool balance deltas from the swap (incoming balance, outgoing balance)
|
|
returns (int256[] memory amounts) {
|
|
// Outgoing balance is negative so we need to flip the sign
|
|
int256 amountOutFromPool = amounts[1] * -1;
|
|
if (amountOutFromPool <= 0) {
|
|
break;
|
|
}
|
|
makerTokenAmounts[i] = uint256(amountOutFromPool);
|
|
} catch (bytes memory) {
|
|
// Swallow failures, leaving all results as zero.
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
/// @dev Sample buy quotes from Balancer V2.
|
|
/// @param poolInfo Struct with pool related data
|
|
/// @param takerToken Address of the taker token (what to sell).
|
|
/// @param makerToken Address of the maker token (what to buy).
|
|
/// @param makerTokenAmounts Maker token buy amount for each sample.
|
|
/// @return takerTokenAmounts Taker amounts sold at each maker token
|
|
/// amount.
|
|
function sampleBuysFromBalancerV2(
|
|
BalancerV2PoolInfo memory poolInfo,
|
|
address takerToken,
|
|
address makerToken,
|
|
uint256[] memory makerTokenAmounts
|
|
)
|
|
public
|
|
returns (uint256[] memory takerTokenAmounts)
|
|
{
|
|
_assertValidPair(makerToken, takerToken);
|
|
IBalancerV2Vault vault = IBalancerV2Vault(poolInfo.vault);
|
|
IAsset[] memory swapAssets = new IAsset[](2);
|
|
swapAssets[0] = IAsset(takerToken);
|
|
swapAssets[1] = IAsset(makerToken);
|
|
|
|
uint256 numSamples = makerTokenAmounts.length;
|
|
takerTokenAmounts = new uint256[](numSamples);
|
|
IBalancerV2Vault.FundManagement memory swapFunds =
|
|
_createSwapFunds();
|
|
|
|
for (uint256 i = 0; i < numSamples; i++) {
|
|
IBalancerV2Vault.BatchSwapStep[] memory swapSteps =
|
|
_createSwapSteps(poolInfo, makerTokenAmounts[i]);
|
|
|
|
try
|
|
// For buys we specify the makerToken which is what taker will receive from the trade
|
|
vault.queryBatchSwap(IBalancerV2Vault.SwapKind.GIVEN_OUT, swapSteps, swapAssets, swapFunds)
|
|
returns (int256[] memory amounts) {
|
|
int256 amountIntoPool = amounts[0];
|
|
if (amountIntoPool <= 0) {
|
|
break;
|
|
}
|
|
takerTokenAmounts[i] = uint256(amountIntoPool);
|
|
} catch (bytes memory) {
|
|
// Swallow failures, leaving all results as zero.
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
function _createSwapSteps(
|
|
BalancerV2PoolInfo memory poolInfo,
|
|
uint256 amount
|
|
) private pure returns (IBalancerV2Vault.BatchSwapStep[] memory) {
|
|
IBalancerV2Vault.BatchSwapStep[] memory swapSteps =
|
|
new IBalancerV2Vault.BatchSwapStep[](1);
|
|
swapSteps[0] = IBalancerV2Vault.BatchSwapStep({
|
|
poolId: poolInfo.poolId,
|
|
assetInIndex: 0,
|
|
assetOutIndex: 1,
|
|
amount: amount,
|
|
userData: ""
|
|
});
|
|
|
|
return swapSteps;
|
|
}
|
|
|
|
function _createSwapFunds()
|
|
private
|
|
view
|
|
returns (IBalancerV2Vault.FundManagement memory)
|
|
{
|
|
return
|
|
IBalancerV2Vault.FundManagement({
|
|
sender: address(this),
|
|
fromInternalBalance: false,
|
|
recipient: payable(address(this)),
|
|
toInternalBalance: false
|
|
});
|
|
}
|
|
}
|