2021-02-04 19:15:23 +01:00
|
|
|
import BigNumber from 'bignumber.js';
|
|
|
|
import abi from 'human-standard-token-abi';
|
2021-03-09 17:06:44 +01:00
|
|
|
import {
|
2021-03-18 11:20:06 +01:00
|
|
|
SWAPS_CHAINID_DEFAULT_TOKEN_MAP,
|
2021-09-08 20:08:55 +02:00
|
|
|
ALLOWED_CONTRACT_ADDRESSES,
|
2021-07-09 17:24:00 +02:00
|
|
|
ETHEREUM,
|
|
|
|
POLYGON,
|
|
|
|
BSC,
|
2022-09-12 13:11:29 +02:00
|
|
|
GOERLI,
|
2022-02-18 15:24:47 +01:00
|
|
|
AVALANCHE,
|
2022-11-08 19:14:17 +01:00
|
|
|
OPTIMISM,
|
|
|
|
ARBITRUM,
|
2021-09-08 19:26:37 +02:00
|
|
|
SWAPS_API_V2_BASE_URL,
|
|
|
|
SWAPS_DEV_API_V2_BASE_URL,
|
2021-09-29 15:11:19 +02:00
|
|
|
SWAPS_CLIENT_ID,
|
2021-04-28 21:53:59 +02:00
|
|
|
} from '../../../shared/constants/swaps';
|
2021-03-18 11:20:06 +01:00
|
|
|
import {
|
|
|
|
isSwapsDefaultTokenAddress,
|
|
|
|
isSwapsDefaultTokenSymbol,
|
2021-04-28 21:53:59 +02:00
|
|
|
} from '../../../shared/modules/swaps.utils';
|
2022-09-14 16:55:31 +02:00
|
|
|
import { CHAIN_IDS, CURRENCY_SYMBOLS } from '../../../shared/constants/network';
|
2022-09-16 21:05:21 +02:00
|
|
|
import { getValueFromWeiHex } from '../../helpers/utils/conversions.util';
|
|
|
|
import { formatCurrency } from '../../helpers/utils/confirm-tx.util';
|
|
|
|
import fetchWithCache from '../../../shared/lib/fetch-with-cache';
|
|
|
|
|
|
|
|
import { isValidHexAddress } from '../../../shared/modules/hexstring-utils';
|
2021-02-04 19:15:23 +01:00
|
|
|
import {
|
2022-09-16 21:05:21 +02:00
|
|
|
calcGasTotal,
|
2021-02-04 19:15:23 +01:00
|
|
|
calcTokenAmount,
|
2022-09-16 21:05:21 +02:00
|
|
|
decimalToHex,
|
2020-11-03 00:41:28 +01:00
|
|
|
toPrecisionWithoutTrailingZeros,
|
2022-09-16 21:05:21 +02:00
|
|
|
} from '../../../shared/lib/transactions-controller-utils';
|
2020-11-03 00:41:28 +01:00
|
|
|
import {
|
2022-09-16 21:05:21 +02:00
|
|
|
getBaseApi,
|
|
|
|
truthyString,
|
|
|
|
validateData,
|
|
|
|
} from '../../../shared/lib/swaps-utils';
|
2022-11-08 19:14:17 +01:00
|
|
|
import { sumHexes } from '../../helpers/utils/transactions.util';
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2021-04-10 02:59:52 +02:00
|
|
|
const CACHE_REFRESH_FIVE_MINUTES = 300000;
|
2022-03-23 20:28:26 +01:00
|
|
|
const USD_CURRENCY_CODE = 'usd';
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2021-09-29 15:11:19 +02:00
|
|
|
const clientIdHeader = { 'X-Client-Id': SWAPS_CLIENT_ID };
|
2021-09-20 21:06:41 +02:00
|
|
|
|
2020-10-06 20:28:38 +02:00
|
|
|
const TOKEN_VALIDATORS = [
|
|
|
|
{
|
|
|
|
property: 'address',
|
|
|
|
type: 'string',
|
2021-05-17 21:00:59 +02:00
|
|
|
validator: (input) => isValidHexAddress(input, { allowNonPrefixed: false }),
|
2020-10-06 20:28:38 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
property: 'symbol',
|
|
|
|
type: 'string',
|
|
|
|
validator: (string) => truthyString(string) && string.length <= 12,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
property: 'decimals',
|
|
|
|
type: 'string|number',
|
|
|
|
validator: (string) => Number(string) >= 0 && Number(string) <= 36,
|
|
|
|
},
|
2021-02-04 19:15:23 +01:00
|
|
|
];
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2021-02-04 19:15:23 +01:00
|
|
|
const TOP_ASSET_VALIDATORS = TOKEN_VALIDATORS.slice(0, 2);
|
2020-10-06 20:28:38 +02:00
|
|
|
|
|
|
|
const AGGREGATOR_METADATA_VALIDATORS = [
|
|
|
|
{
|
|
|
|
property: 'color',
|
|
|
|
type: 'string',
|
|
|
|
validator: (string) => Boolean(string.match(/^#[A-Fa-f0-9]+$/u)),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
property: 'title',
|
|
|
|
type: 'string',
|
|
|
|
validator: truthyString,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
property: 'icon',
|
|
|
|
type: 'string',
|
|
|
|
validator: (string) => Boolean(string.match(/^data:image/u)),
|
|
|
|
},
|
2021-02-04 19:15:23 +01:00
|
|
|
];
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2020-11-04 17:14:08 +01:00
|
|
|
const isValidDecimalNumber = (string) =>
|
2021-02-04 19:15:23 +01:00
|
|
|
!isNaN(string) && string.match(/^[.0-9]+$/u) && !isNaN(parseFloat(string));
|
2020-11-04 17:14:08 +01:00
|
|
|
|
|
|
|
const SWAP_GAS_PRICE_VALIDATOR = [
|
|
|
|
{
|
|
|
|
property: 'SafeGasPrice',
|
|
|
|
type: 'string',
|
|
|
|
validator: isValidDecimalNumber,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
property: 'ProposeGasPrice',
|
|
|
|
type: 'string',
|
|
|
|
validator: isValidDecimalNumber,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
property: 'FastGasPrice',
|
|
|
|
type: 'string',
|
|
|
|
validator: isValidDecimalNumber,
|
|
|
|
},
|
2021-02-04 19:15:23 +01:00
|
|
|
];
|
2020-11-04 17:14:08 +01:00
|
|
|
|
2021-11-30 19:56:28 +01:00
|
|
|
export async function fetchToken(contractAddress, chainId) {
|
|
|
|
const tokenUrl = getBaseApi('token', chainId);
|
2021-06-03 18:08:37 +02:00
|
|
|
const token = await fetchWithCache(
|
|
|
|
`${tokenUrl}?address=${contractAddress}`,
|
2021-09-20 21:06:41 +02:00
|
|
|
{ method: 'GET', headers: clientIdHeader },
|
2021-06-03 18:08:37 +02:00
|
|
|
{ cacheRefreshTime: CACHE_REFRESH_FIVE_MINUTES },
|
|
|
|
);
|
|
|
|
return token;
|
|
|
|
}
|
|
|
|
|
2021-11-30 19:56:28 +01:00
|
|
|
export async function fetchTokens(chainId) {
|
|
|
|
const tokensUrl = getBaseApi('tokens', chainId);
|
2020-11-03 00:41:28 +01:00
|
|
|
const tokens = await fetchWithCache(
|
2021-06-03 18:08:37 +02:00
|
|
|
tokensUrl,
|
2021-09-20 21:06:41 +02:00
|
|
|
{ method: 'GET', headers: clientIdHeader },
|
2021-04-10 02:59:52 +02:00
|
|
|
{ cacheRefreshTime: CACHE_REFRESH_FIVE_MINUTES },
|
2021-02-04 19:15:23 +01:00
|
|
|
);
|
2022-07-28 15:56:15 +02:00
|
|
|
const logError = false;
|
2021-03-15 17:51:43 +01:00
|
|
|
const filteredTokens = [
|
2021-03-18 11:20:06 +01:00
|
|
|
SWAPS_CHAINID_DEFAULT_TOKEN_MAP[chainId],
|
2021-03-15 17:51:43 +01:00
|
|
|
...tokens.filter((token) => {
|
|
|
|
return (
|
2022-07-28 15:56:15 +02:00
|
|
|
validateData(TOKEN_VALIDATORS, token, tokensUrl, logError) &&
|
2021-03-15 17:51:43 +01:00
|
|
|
!(
|
2021-03-18 11:20:06 +01:00
|
|
|
isSwapsDefaultTokenSymbol(token.symbol, chainId) ||
|
|
|
|
isSwapsDefaultTokenAddress(token.address, chainId)
|
2021-03-15 17:51:43 +01:00
|
|
|
)
|
|
|
|
);
|
|
|
|
}),
|
|
|
|
];
|
2021-02-04 19:15:23 +01:00
|
|
|
return filteredTokens;
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
|
2021-11-30 19:56:28 +01:00
|
|
|
export async function fetchAggregatorMetadata(chainId) {
|
|
|
|
const aggregatorMetadataUrl = getBaseApi('aggregatorMetadata', chainId);
|
2020-11-03 00:41:28 +01:00
|
|
|
const aggregators = await fetchWithCache(
|
|
|
|
aggregatorMetadataUrl,
|
2021-09-20 21:06:41 +02:00
|
|
|
{ method: 'GET', headers: clientIdHeader },
|
2021-04-10 02:59:52 +02:00
|
|
|
{ cacheRefreshTime: CACHE_REFRESH_FIVE_MINUTES },
|
2021-02-04 19:15:23 +01:00
|
|
|
);
|
|
|
|
const filteredAggregators = {};
|
2020-10-06 20:28:38 +02:00
|
|
|
for (const aggKey in aggregators) {
|
2020-11-03 00:41:28 +01:00
|
|
|
if (
|
|
|
|
validateData(
|
|
|
|
AGGREGATOR_METADATA_VALIDATORS,
|
|
|
|
aggregators[aggKey],
|
|
|
|
aggregatorMetadataUrl,
|
|
|
|
)
|
|
|
|
) {
|
2021-02-04 19:15:23 +01:00
|
|
|
filteredAggregators[aggKey] = aggregators[aggKey];
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
}
|
2021-02-04 19:15:23 +01:00
|
|
|
return filteredAggregators;
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
|
2021-11-30 19:56:28 +01:00
|
|
|
export async function fetchTopAssets(chainId) {
|
|
|
|
const topAssetsUrl = getBaseApi('topAssets', chainId);
|
2022-05-24 18:30:46 +02:00
|
|
|
const response =
|
|
|
|
(await fetchWithCache(
|
|
|
|
topAssetsUrl,
|
|
|
|
{ method: 'GET', headers: clientIdHeader },
|
|
|
|
{ cacheRefreshTime: CACHE_REFRESH_FIVE_MINUTES },
|
|
|
|
)) || [];
|
2020-10-06 20:28:38 +02:00
|
|
|
const topAssetsMap = response.reduce((_topAssetsMap, asset, index) => {
|
|
|
|
if (validateData(TOP_ASSET_VALIDATORS, asset, topAssetsUrl)) {
|
2021-02-04 19:15:23 +01:00
|
|
|
return { ..._topAssetsMap, [asset.address]: { index: String(index) } };
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
2021-02-04 19:15:23 +01:00
|
|
|
return _topAssetsMap;
|
|
|
|
}, {});
|
|
|
|
return topAssetsMap;
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
|
2021-07-09 17:24:00 +02:00
|
|
|
export async function fetchSwapsFeatureFlags() {
|
2021-09-08 19:26:37 +02:00
|
|
|
const v2ApiBaseUrl = process.env.SWAPS_USE_DEV_APIS
|
|
|
|
? SWAPS_DEV_API_V2_BASE_URL
|
|
|
|
: SWAPS_API_V2_BASE_URL;
|
2021-07-09 17:24:00 +02:00
|
|
|
const response = await fetchWithCache(
|
2021-09-08 19:26:37 +02:00
|
|
|
`${v2ApiBaseUrl}/featureFlags`,
|
2021-09-20 21:06:41 +02:00
|
|
|
{ method: 'GET', headers: clientIdHeader },
|
2020-11-03 00:41:28 +01:00
|
|
|
{ cacheRefreshTime: 600000 },
|
2021-02-04 19:15:23 +01:00
|
|
|
);
|
2021-07-09 17:24:00 +02:00
|
|
|
return response;
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
export async function fetchTokenPrice(address) {
|
2021-02-04 19:15:23 +01:00
|
|
|
const query = `contract_addresses=${address}&vs_currencies=eth`;
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
const prices = await fetchWithCache(
|
|
|
|
`https://api.coingecko.com/api/v3/simple/token_price/ethereum?${query}`,
|
|
|
|
{ method: 'GET' },
|
|
|
|
{ cacheRefreshTime: 60000 },
|
2021-02-04 19:15:23 +01:00
|
|
|
);
|
|
|
|
return prices && prices[address]?.eth;
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
export async function fetchTokenBalance(address, userAddress) {
|
2021-02-04 19:15:23 +01:00
|
|
|
const tokenContract = global.eth.contract(abi).at(address);
|
2020-10-06 20:28:38 +02:00
|
|
|
const tokenBalancePromise = tokenContract
|
|
|
|
? tokenContract.balanceOf(userAddress)
|
2021-02-04 19:15:23 +01:00
|
|
|
: Promise.resolve();
|
|
|
|
const usersToken = await tokenBalancePromise;
|
|
|
|
return usersToken;
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
|
2021-11-30 19:56:28 +01:00
|
|
|
export async function fetchSwapsGasPrices(chainId) {
|
|
|
|
const gasPricesUrl = getBaseApi('gasPrices', chainId);
|
2020-11-04 17:14:08 +01:00
|
|
|
const response = await fetchWithCache(
|
|
|
|
gasPricesUrl,
|
2021-09-20 21:06:41 +02:00
|
|
|
{ method: 'GET', headers: clientIdHeader },
|
2021-03-28 15:18:44 +02:00
|
|
|
{ cacheRefreshTime: 30000 },
|
2021-02-04 19:15:23 +01:00
|
|
|
);
|
2020-11-04 17:14:08 +01:00
|
|
|
const responseIsValid = validateData(
|
|
|
|
SWAP_GAS_PRICE_VALIDATOR,
|
|
|
|
response,
|
|
|
|
gasPricesUrl,
|
2021-02-04 19:15:23 +01:00
|
|
|
);
|
2020-11-04 17:14:08 +01:00
|
|
|
|
|
|
|
if (!responseIsValid) {
|
2021-02-04 19:15:23 +01:00
|
|
|
throw new Error(`${gasPricesUrl} response is invalid`);
|
2020-11-04 17:14:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
const {
|
|
|
|
SafeGasPrice: safeLow,
|
|
|
|
ProposeGasPrice: average,
|
|
|
|
FastGasPrice: fast,
|
2021-02-04 19:15:23 +01:00
|
|
|
} = response;
|
2020-11-04 17:14:08 +01:00
|
|
|
|
|
|
|
return {
|
|
|
|
safeLow,
|
|
|
|
average,
|
|
|
|
fast,
|
2021-02-04 19:15:23 +01:00
|
|
|
};
|
2020-11-04 17:14:08 +01:00
|
|
|
}
|
|
|
|
|
2022-02-18 17:48:38 +01:00
|
|
|
export const getFeeForSmartTransaction = ({
|
|
|
|
chainId,
|
|
|
|
currentCurrency,
|
|
|
|
conversionRate,
|
2022-05-09 18:48:14 +02:00
|
|
|
USDConversionRate,
|
2022-02-18 17:48:38 +01:00
|
|
|
nativeCurrencySymbol,
|
|
|
|
feeInWeiDec,
|
|
|
|
}) => {
|
|
|
|
const feeInWeiHex = decimalToHex(feeInWeiDec);
|
|
|
|
const ethFee = getValueFromWeiHex({
|
|
|
|
value: feeInWeiHex,
|
2022-09-14 16:55:31 +02:00
|
|
|
toDenomination: CURRENCY_SYMBOLS.ETH,
|
2022-02-18 17:48:38 +01:00
|
|
|
numberOfDecimals: 5,
|
|
|
|
});
|
|
|
|
const rawNetworkFees = getValueFromWeiHex({
|
|
|
|
value: feeInWeiHex,
|
|
|
|
toCurrency: currentCurrency,
|
|
|
|
conversionRate,
|
|
|
|
numberOfDecimals: 2,
|
|
|
|
});
|
2022-03-23 20:28:26 +01:00
|
|
|
let feeInUsd;
|
|
|
|
if (currentCurrency === USD_CURRENCY_CODE) {
|
|
|
|
feeInUsd = rawNetworkFees;
|
|
|
|
} else {
|
|
|
|
feeInUsd = getValueFromWeiHex({
|
|
|
|
value: feeInWeiHex,
|
|
|
|
toCurrency: USD_CURRENCY_CODE,
|
2022-05-09 18:48:14 +02:00
|
|
|
conversionRate: USDConversionRate,
|
2022-03-23 20:28:26 +01:00
|
|
|
numberOfDecimals: 2,
|
|
|
|
});
|
|
|
|
}
|
2022-02-18 17:48:38 +01:00
|
|
|
const formattedNetworkFee = formatCurrency(rawNetworkFees, currentCurrency);
|
|
|
|
const chainCurrencySymbolToUse =
|
|
|
|
nativeCurrencySymbol || SWAPS_CHAINID_DEFAULT_TOKEN_MAP[chainId].symbol;
|
|
|
|
return {
|
2022-03-23 20:28:26 +01:00
|
|
|
feeInUsd,
|
2022-02-18 17:48:38 +01:00
|
|
|
feeInFiat: formattedNetworkFee,
|
|
|
|
feeInEth: `${ethFee} ${chainCurrencySymbolToUse}`,
|
2022-03-23 20:28:26 +01:00
|
|
|
rawEthFee: ethFee,
|
2022-02-18 17:48:38 +01:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2021-03-18 11:20:06 +01:00
|
|
|
export function getRenderableNetworkFeesForQuote({
|
2020-10-27 20:02:21 +01:00
|
|
|
tradeGas,
|
|
|
|
approveGas,
|
|
|
|
gasPrice,
|
|
|
|
currentCurrency,
|
|
|
|
conversionRate,
|
2022-05-09 18:48:14 +02:00
|
|
|
USDConversionRate,
|
2020-10-27 20:02:21 +01:00
|
|
|
tradeValue,
|
|
|
|
sourceSymbol,
|
|
|
|
sourceAmount,
|
2021-03-18 11:20:06 +01:00
|
|
|
chainId,
|
2021-03-29 21:30:06 +02:00
|
|
|
nativeCurrencySymbol,
|
2022-11-08 19:14:17 +01:00
|
|
|
multiLayerL1FeeTotal,
|
2021-03-18 11:20:06 +01:00
|
|
|
}) {
|
2020-11-03 00:41:28 +01:00
|
|
|
const totalGasLimitForCalculation = new BigNumber(tradeGas || '0x0', 16)
|
|
|
|
.plus(approveGas || '0x0', 16)
|
2021-02-04 19:15:23 +01:00
|
|
|
.toString(16);
|
2022-11-08 19:14:17 +01:00
|
|
|
let gasTotalInWeiHex = calcGasTotal(totalGasLimitForCalculation, gasPrice);
|
|
|
|
if (multiLayerL1FeeTotal !== null) {
|
|
|
|
gasTotalInWeiHex = sumHexes(
|
|
|
|
gasTotalInWeiHex || '0x0',
|
|
|
|
multiLayerL1FeeTotal || '0x0',
|
|
|
|
);
|
|
|
|
}
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2020-10-27 20:02:21 +01:00
|
|
|
const nonGasFee = new BigNumber(tradeValue, 16)
|
2021-03-18 11:20:06 +01:00
|
|
|
.minus(
|
|
|
|
isSwapsDefaultTokenSymbol(sourceSymbol, chainId) ? sourceAmount : 0,
|
|
|
|
10,
|
|
|
|
)
|
2021-02-04 19:15:23 +01:00
|
|
|
.toString(16);
|
2020-10-27 20:02:21 +01:00
|
|
|
|
|
|
|
const totalWeiCost = new BigNumber(gasTotalInWeiHex, 16)
|
|
|
|
.plus(nonGasFee, 16)
|
2021-02-04 19:15:23 +01:00
|
|
|
.toString(16);
|
2020-10-27 20:02:21 +01:00
|
|
|
|
2020-10-06 20:28:38 +02:00
|
|
|
const ethFee = getValueFromWeiHex({
|
2020-10-27 20:02:21 +01:00
|
|
|
value: totalWeiCost,
|
2020-10-06 20:28:38 +02:00
|
|
|
toDenomination: 'ETH',
|
2020-10-20 01:29:31 +02:00
|
|
|
numberOfDecimals: 5,
|
2021-02-04 19:15:23 +01:00
|
|
|
});
|
2020-10-06 20:28:38 +02:00
|
|
|
const rawNetworkFees = getValueFromWeiHex({
|
2020-10-27 20:02:21 +01:00
|
|
|
value: totalWeiCost,
|
2020-10-06 20:28:38 +02:00
|
|
|
toCurrency: currentCurrency,
|
|
|
|
conversionRate,
|
|
|
|
numberOfDecimals: 2,
|
2021-02-04 19:15:23 +01:00
|
|
|
});
|
|
|
|
const formattedNetworkFee = formatCurrency(rawNetworkFees, currentCurrency);
|
2021-03-29 21:30:06 +02:00
|
|
|
|
2022-03-23 20:28:26 +01:00
|
|
|
let feeInUsd;
|
|
|
|
if (currentCurrency === USD_CURRENCY_CODE) {
|
|
|
|
feeInUsd = rawNetworkFees;
|
|
|
|
} else {
|
|
|
|
feeInUsd = getValueFromWeiHex({
|
|
|
|
value: totalWeiCost,
|
|
|
|
toCurrency: USD_CURRENCY_CODE,
|
2022-05-09 18:48:14 +02:00
|
|
|
conversionRate: USDConversionRate,
|
2022-03-23 20:28:26 +01:00
|
|
|
numberOfDecimals: 2,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-03-29 21:30:06 +02:00
|
|
|
const chainCurrencySymbolToUse =
|
|
|
|
nativeCurrencySymbol || SWAPS_CHAINID_DEFAULT_TOKEN_MAP[chainId].symbol;
|
|
|
|
|
2020-10-06 20:28:38 +02:00
|
|
|
return {
|
|
|
|
rawNetworkFees,
|
2022-03-23 20:28:26 +01:00
|
|
|
feeInUsd,
|
2020-10-06 20:28:38 +02:00
|
|
|
rawEthFee: ethFee,
|
|
|
|
feeInFiat: formattedNetworkFee,
|
2021-03-29 21:30:06 +02:00
|
|
|
feeInEth: `${ethFee} ${chainCurrencySymbolToUse}`,
|
2020-10-27 20:02:21 +01:00
|
|
|
nonGasFee,
|
2021-02-04 19:15:23 +01:00
|
|
|
};
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
export function quotesToRenderableData(
|
|
|
|
quotes,
|
|
|
|
gasPrice,
|
|
|
|
conversionRate,
|
|
|
|
currentCurrency,
|
|
|
|
approveGas,
|
|
|
|
tokenConversionRates,
|
2021-03-18 11:20:06 +01:00
|
|
|
chainId,
|
2022-02-18 17:48:38 +01:00
|
|
|
smartTransactionEstimatedGas,
|
|
|
|
nativeCurrencySymbol,
|
2022-12-21 20:59:34 +01:00
|
|
|
multiLayerL1ApprovalFeeTotal,
|
2020-11-03 00:41:28 +01:00
|
|
|
) {
|
2020-10-06 20:28:38 +02:00
|
|
|
return Object.values(quotes).map((quote) => {
|
2020-10-22 16:58:19 +02:00
|
|
|
const {
|
|
|
|
destinationAmount = 0,
|
|
|
|
sourceAmount = 0,
|
|
|
|
sourceTokenInfo,
|
|
|
|
destinationTokenInfo,
|
|
|
|
slippage,
|
|
|
|
aggType,
|
|
|
|
aggregator,
|
|
|
|
gasEstimateWithRefund,
|
|
|
|
averageGas,
|
|
|
|
fee,
|
2020-10-27 20:02:21 +01:00
|
|
|
trade,
|
2022-12-21 20:59:34 +01:00
|
|
|
multiLayerL1TradeFeeTotal,
|
2021-02-04 19:15:23 +01:00
|
|
|
} = quote;
|
2022-12-21 20:59:34 +01:00
|
|
|
let multiLayerL1FeeTotal = null;
|
|
|
|
if (
|
|
|
|
multiLayerL1TradeFeeTotal !== null &&
|
|
|
|
multiLayerL1ApprovalFeeTotal !== null
|
|
|
|
) {
|
|
|
|
multiLayerL1FeeTotal = sumHexes(
|
|
|
|
multiLayerL1TradeFeeTotal || '0x0',
|
|
|
|
multiLayerL1ApprovalFeeTotal || '0x0',
|
|
|
|
);
|
|
|
|
} else if (multiLayerL1TradeFeeTotal !== null) {
|
|
|
|
multiLayerL1FeeTotal = multiLayerL1TradeFeeTotal;
|
|
|
|
}
|
2020-11-03 00:41:28 +01:00
|
|
|
const sourceValue = calcTokenAmount(
|
|
|
|
sourceAmount,
|
2021-01-19 18:05:58 +01:00
|
|
|
sourceTokenInfo.decimals,
|
2021-02-04 19:15:23 +01:00
|
|
|
).toString(10);
|
2020-11-03 00:41:28 +01:00
|
|
|
const destinationValue = calcTokenAmount(
|
|
|
|
destinationAmount,
|
2021-01-19 18:05:58 +01:00
|
|
|
destinationTokenInfo.decimals,
|
2021-02-04 19:15:23 +01:00
|
|
|
).toPrecision(8);
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2022-02-18 17:48:38 +01:00
|
|
|
let feeInFiat = null;
|
|
|
|
let feeInEth = null;
|
|
|
|
let rawNetworkFees = null;
|
|
|
|
let rawEthFee = null;
|
|
|
|
|
2022-07-31 20:26:40 +02:00
|
|
|
({ feeInFiat, feeInEth, rawNetworkFees, rawEthFee } =
|
|
|
|
getRenderableNetworkFeesForQuote({
|
|
|
|
tradeGas: gasEstimateWithRefund || decimalToHex(averageGas || 800000),
|
|
|
|
approveGas,
|
|
|
|
gasPrice,
|
|
|
|
currentCurrency,
|
|
|
|
conversionRate,
|
|
|
|
tradeValue: trade.value,
|
|
|
|
sourceSymbol: sourceTokenInfo.symbol,
|
|
|
|
sourceAmount,
|
|
|
|
chainId,
|
2022-11-08 19:14:17 +01:00
|
|
|
multiLayerL1FeeTotal,
|
2022-07-31 20:26:40 +02:00
|
|
|
}));
|
2022-02-18 17:48:38 +01:00
|
|
|
|
|
|
|
if (smartTransactionEstimatedGas) {
|
|
|
|
({ feeInFiat, feeInEth } = getFeeForSmartTransaction({
|
|
|
|
chainId,
|
|
|
|
currentCurrency,
|
|
|
|
conversionRate,
|
|
|
|
nativeCurrencySymbol,
|
|
|
|
estimatedFeeInWeiDec: smartTransactionEstimatedGas.feeEstimate,
|
|
|
|
}));
|
|
|
|
}
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2021-02-04 19:15:23 +01:00
|
|
|
const slippageMultiplier = new BigNumber(100 - slippage).div(100);
|
2020-11-03 00:41:28 +01:00
|
|
|
const minimumAmountReceived = new BigNumber(destinationValue)
|
|
|
|
.times(slippageMultiplier)
|
2021-02-04 19:15:23 +01:00
|
|
|
.toFixed(6);
|
2020-11-03 00:41:28 +01:00
|
|
|
|
|
|
|
const tokenConversionRate =
|
2021-02-04 19:15:23 +01:00
|
|
|
tokenConversionRates[destinationTokenInfo.address];
|
2021-03-18 11:20:06 +01:00
|
|
|
const ethValueOfTrade = isSwapsDefaultTokenSymbol(
|
|
|
|
destinationTokenInfo.symbol,
|
|
|
|
chainId,
|
|
|
|
)
|
|
|
|
? calcTokenAmount(destinationAmount, destinationTokenInfo.decimals).minus(
|
|
|
|
rawEthFee,
|
|
|
|
10,
|
|
|
|
)
|
|
|
|
: new BigNumber(tokenConversionRate || 0, 10)
|
|
|
|
.times(
|
|
|
|
calcTokenAmount(destinationAmount, destinationTokenInfo.decimals),
|
|
|
|
10,
|
|
|
|
)
|
|
|
|
.minus(rawEthFee, 10);
|
2020-10-06 20:28:38 +02:00
|
|
|
|
2021-02-04 19:15:23 +01:00
|
|
|
let liquiditySourceKey;
|
|
|
|
let renderedSlippage = slippage;
|
2020-10-06 20:28:38 +02:00
|
|
|
|
|
|
|
if (aggType === 'AGG') {
|
2021-02-04 19:15:23 +01:00
|
|
|
liquiditySourceKey = 'swapAggregator';
|
2020-10-06 20:28:38 +02:00
|
|
|
} else if (aggType === 'RFQ') {
|
2021-02-04 19:15:23 +01:00
|
|
|
liquiditySourceKey = 'swapRequestForQuotation';
|
|
|
|
renderedSlippage = 0;
|
2020-10-06 20:28:38 +02:00
|
|
|
} else if (aggType === 'DEX') {
|
2021-02-04 19:15:23 +01:00
|
|
|
liquiditySourceKey = 'swapDecentralizedExchange';
|
2021-09-08 20:08:55 +02:00
|
|
|
} else if (aggType === 'CONTRACT') {
|
|
|
|
liquiditySourceKey = 'swapDirectContract';
|
2020-10-06 20:28:38 +02:00
|
|
|
} else {
|
2021-02-04 19:15:23 +01:00
|
|
|
liquiditySourceKey = 'swapUnknown';
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
aggId: aggregator,
|
|
|
|
amountReceiving: `${destinationValue} ${destinationTokenInfo.symbol}`,
|
|
|
|
destinationTokenDecimals: destinationTokenInfo.decimals,
|
|
|
|
destinationTokenSymbol: destinationTokenInfo.symbol,
|
2020-10-07 02:37:38 +02:00
|
|
|
destinationTokenValue: formatSwapsValueForDisplay(destinationValue),
|
2020-12-08 17:47:53 +01:00
|
|
|
destinationIconUrl: destinationTokenInfo.iconUrl,
|
2020-10-06 20:28:38 +02:00
|
|
|
isBestQuote: quote.isBestQuote,
|
|
|
|
liquiditySourceKey,
|
|
|
|
feeInEth,
|
|
|
|
detailedNetworkFees: `${feeInEth} (${feeInFiat})`,
|
|
|
|
networkFees: feeInFiat,
|
|
|
|
quoteSource: aggType,
|
|
|
|
rawNetworkFees,
|
|
|
|
slippage: renderedSlippage,
|
|
|
|
sourceTokenDecimals: sourceTokenInfo.decimals,
|
|
|
|
sourceTokenSymbol: sourceTokenInfo.symbol,
|
|
|
|
sourceTokenValue: sourceValue,
|
2020-12-08 17:47:53 +01:00
|
|
|
sourceTokenIconUrl: sourceTokenInfo.iconUrl,
|
2020-10-06 20:28:38 +02:00
|
|
|
ethValueOfTrade,
|
|
|
|
minimumAmountReceived,
|
2020-10-22 16:58:19 +02:00
|
|
|
metaMaskFee: fee,
|
2021-02-04 19:15:23 +01:00
|
|
|
};
|
|
|
|
});
|
2020-10-06 20:28:38 +02:00
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
export function formatSwapsValueForDisplay(destinationAmount) {
|
2021-02-04 19:15:23 +01:00
|
|
|
let amountToDisplay = toPrecisionWithoutTrailingZeros(destinationAmount, 12);
|
2020-10-07 02:37:38 +02:00
|
|
|
if (amountToDisplay.match(/e[+-]/u)) {
|
2021-02-04 19:15:23 +01:00
|
|
|
amountToDisplay = new BigNumber(amountToDisplay).toFixed();
|
2020-10-07 02:37:38 +02:00
|
|
|
}
|
2021-02-04 19:15:23 +01:00
|
|
|
return amountToDisplay;
|
2020-10-07 02:37:38 +02:00
|
|
|
}
|
2021-04-23 16:53:10 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks whether a contract address is valid before swapping tokens.
|
|
|
|
*
|
|
|
|
* @param {string} contractAddress - E.g. "0x881d40237659c251811cec9c364ef91dc08d300c" for mainnet
|
|
|
|
* @param {string} chainId - The hex encoded chain ID to check
|
|
|
|
* @returns {boolean} Whether a contract address is valid or not
|
|
|
|
*/
|
|
|
|
export const isContractAddressValid = (
|
|
|
|
contractAddress,
|
2022-09-14 16:55:31 +02:00
|
|
|
chainId = CHAIN_IDS.MAINNET,
|
2021-04-23 16:53:10 +02:00
|
|
|
) => {
|
2021-09-08 20:08:55 +02:00
|
|
|
if (!contractAddress || !ALLOWED_CONTRACT_ADDRESSES[chainId]) {
|
2021-04-23 16:53:10 +02:00
|
|
|
return false;
|
|
|
|
}
|
2021-09-08 20:08:55 +02:00
|
|
|
return ALLOWED_CONTRACT_ADDRESSES[chainId].some(
|
2021-04-23 16:53:10 +02:00
|
|
|
// Sometimes we get a contract address with a few upper-case chars and since addresses are
|
2021-09-08 20:08:55 +02:00
|
|
|
// case-insensitive, we compare lowercase versions for validity.
|
|
|
|
(allowedContractAddress) =>
|
|
|
|
contractAddress.toLowerCase() === allowedContractAddress.toLowerCase(),
|
2021-04-23 16:53:10 +02:00
|
|
|
);
|
|
|
|
};
|
2021-07-09 17:24:00 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} chainId
|
|
|
|
* @returns string e.g. ethereum, bsc or polygon
|
|
|
|
*/
|
|
|
|
export const getNetworkNameByChainId = (chainId) => {
|
|
|
|
switch (chainId) {
|
2022-09-14 16:55:31 +02:00
|
|
|
case CHAIN_IDS.MAINNET:
|
2021-07-09 17:24:00 +02:00
|
|
|
return ETHEREUM;
|
2022-09-14 16:55:31 +02:00
|
|
|
case CHAIN_IDS.BSC:
|
2021-07-09 17:24:00 +02:00
|
|
|
return BSC;
|
2022-09-14 16:55:31 +02:00
|
|
|
case CHAIN_IDS.POLYGON:
|
2021-07-09 17:24:00 +02:00
|
|
|
return POLYGON;
|
2022-09-14 16:55:31 +02:00
|
|
|
case CHAIN_IDS.GOERLI:
|
2022-09-12 13:11:29 +02:00
|
|
|
return GOERLI;
|
2022-09-14 16:55:31 +02:00
|
|
|
case CHAIN_IDS.AVALANCHE:
|
2022-02-18 15:24:47 +01:00
|
|
|
return AVALANCHE;
|
2022-11-08 19:14:17 +01:00
|
|
|
case CHAIN_IDS.OPTIMISM:
|
|
|
|
return OPTIMISM;
|
|
|
|
case CHAIN_IDS.ARBITRUM:
|
|
|
|
return ARBITRUM;
|
2021-07-09 17:24:00 +02:00
|
|
|
default:
|
|
|
|
return '';
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* It returns info about if Swaps are enabled and if we should use our new APIs for it.
|
2022-01-07 16:57:33 +01:00
|
|
|
*
|
2021-07-09 17:24:00 +02:00
|
|
|
* @param {object} swapsFeatureFlags
|
|
|
|
* @param {string} chainId
|
2021-11-30 19:56:28 +01:00
|
|
|
* @returns object with 2 items: "swapsFeatureIsLive"
|
2021-07-09 17:24:00 +02:00
|
|
|
*/
|
|
|
|
export const getSwapsLivenessForNetwork = (swapsFeatureFlags = {}, chainId) => {
|
|
|
|
const networkName = getNetworkNameByChainId(chainId);
|
2022-09-29 05:26:01 +02:00
|
|
|
// Use old APIs for testnet and Goerli.
|
2022-09-14 16:55:31 +02:00
|
|
|
if ([CHAIN_IDS.LOCALHOST, CHAIN_IDS.GOERLI].includes(chainId)) {
|
2021-07-09 17:24:00 +02:00
|
|
|
return {
|
|
|
|
swapsFeatureIsLive: true,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
// If a network name is not found in the list of feature flags, disable Swaps.
|
|
|
|
if (!swapsFeatureFlags[networkName]) {
|
|
|
|
return {
|
|
|
|
swapsFeatureIsLive: false,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
const isNetworkEnabledForNewApi =
|
2022-09-29 18:03:38 +02:00
|
|
|
swapsFeatureFlags[networkName].extensionActive;
|
2021-07-09 17:24:00 +02:00
|
|
|
if (isNetworkEnabledForNewApi) {
|
|
|
|
return {
|
|
|
|
swapsFeatureIsLive: true,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
return {
|
2022-09-29 18:03:38 +02:00
|
|
|
swapsFeatureIsLive: swapsFeatureFlags[networkName].fallbackToV1,
|
2021-07-09 17:24:00 +02:00
|
|
|
};
|
|
|
|
};
|
2021-07-22 19:29:31 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {number} value
|
|
|
|
* @returns number
|
|
|
|
*/
|
|
|
|
export const countDecimals = (value) => {
|
2022-01-06 23:56:51 +01:00
|
|
|
if (!value || Math.floor(value) === value) {
|
|
|
|
return 0;
|
|
|
|
}
|
2021-07-22 19:29:31 +02:00
|
|
|
return value.toString().split('.')[1]?.length || 0;
|
|
|
|
};
|
2022-02-18 17:48:38 +01:00
|
|
|
|
|
|
|
export const showRemainingTimeInMinAndSec = (remainingTimeInSec) => {
|
|
|
|
if (!Number.isInteger(remainingTimeInSec)) {
|
|
|
|
return '0:00';
|
|
|
|
}
|
|
|
|
const minutes = Math.floor(remainingTimeInSec / 60);
|
|
|
|
const seconds = remainingTimeInSec % 60;
|
|
|
|
return `${minutes}:${seconds.toString().padStart(2, '0')}`;
|
|
|
|
};
|
|
|
|
|
2022-03-23 20:28:26 +01:00
|
|
|
export const stxErrorTypes = {
|
|
|
|
UNAVAILABLE: 'unavailable',
|
|
|
|
NOT_ENOUGH_FUNDS: 'not_enough_funds',
|
2022-05-03 17:55:41 +02:00
|
|
|
REGULAR_TX_IN_PROGRESS: 'regular_tx_pending',
|
2022-02-18 17:48:38 +01:00
|
|
|
};
|
|
|
|
|
2022-03-23 20:28:26 +01:00
|
|
|
export const getTranslatedStxErrorMessage = (errorType, t) => {
|
|
|
|
switch (errorType) {
|
|
|
|
case stxErrorTypes.UNAVAILABLE:
|
2022-05-03 17:55:41 +02:00
|
|
|
case stxErrorTypes.REGULAR_TX_IN_PROGRESS:
|
2022-03-23 20:28:26 +01:00
|
|
|
return t('stxErrorUnavailable');
|
|
|
|
case stxErrorTypes.NOT_ENOUGH_FUNDS:
|
|
|
|
return t('stxErrorNotEnoughFunds');
|
|
|
|
default:
|
|
|
|
return t('stxErrorUnavailable');
|
|
|
|
}
|
2022-02-18 17:48:38 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
export const parseSmartTransactionsError = (errorMessage) => {
|
|
|
|
const errorJson = errorMessage.slice(12);
|
|
|
|
return JSON.parse(errorJson.trim());
|
|
|
|
};
|