import { ethers } from 'ethers'
import log from 'loglevel'
import BigNumber from 'bignumber.js'
import ObservableStore from 'obs-store'
import { mapValues } from 'lodash'
import ethUtil from 'ethereumjs-util'
import abi from 'human-standard-token-abi'
import { calcTokenAmount } from '../../../ui/app/helpers/utils/token-util'
import { calcGasTotal } from '../../../ui/app/pages/send/send.utils'
import { conversionUtil } from '../../../ui/app/helpers/utils/conversion-util'
import {
ETH _SWAPS _TOKEN _ADDRESS ,
DEFAULT _ERC20 _APPROVE _GAS ,
QUOTES _EXPIRED _ERROR ,
QUOTES _NOT _AVAILABLE _ERROR ,
} from '../../../ui/app/helpers/constants/swaps'
import {
fetchTradesInfo as defaultFetchTradesInfo ,
fetchSwapsFeatureLiveness as defaultFetchSwapsFeatureLiveness ,
} from '../../../ui/app/pages/swaps/swaps.util'
const METASWAP _ADDRESS = '0x016B4bf68d421147c06f1b8680602c5bf0Df91A8'
// The MAX_GAS_LIMIT is a number that is higher than the maximum gas costs we have observed on any aggregator
const MAX _GAS _LIMIT = 2500000
// To ensure that our serves are not spammed if MetaMask is left idle, we limit the number of fetches for quotes that are made on timed intervals.
// 3 seems to be an appropriate balance of giving users the time they need when MetaMask is not left idle, and turning polling off when it is.
const POLL _COUNT _LIMIT = 3
function calculateGasEstimateWithRefund ( maxGas = MAX _GAS _LIMIT , estimatedRefund = 0 , estimatedGas = 0 ) {
const maxGasMinusRefund = new BigNumber (
maxGas ,
10 ,
)
. minus ( estimatedRefund , 10 )
const gasEstimateWithRefund = maxGasMinusRefund . lt (
estimatedGas ,
16 ,
)
? maxGasMinusRefund . toString ( 16 )
: estimatedGas
return gasEstimateWithRefund
}
// This is the amount of time to wait, after successfully fetching quotes and their gas estimates, before fetching for new quotes
const QUOTE _POLLING _INTERVAL = 50 * 1000
const initialState = {
swapsState : {
quotes : { } ,
fetchParams : null ,
tokens : null ,
tradeTxId : null ,
approveTxId : null ,
maxMode : false ,
quotesLastFetched : null ,
customMaxGas : '' ,
customGasPrice : null ,
selectedAggId : null ,
customApproveTxData : '' ,
errorKey : '' ,
topAggId : null ,
routeState : '' ,
swapsFeatureIsLive : false ,
} ,
}
export default class SwapsController {
constructor ( {
getBufferedGasLimit ,
provider ,
getProviderConfig ,
tokenRatesStore ,
fetchTradesInfo = defaultFetchTradesInfo ,
fetchSwapsFeatureLiveness = defaultFetchSwapsFeatureLiveness ,
} ) {
this . store = new ObservableStore ( {
swapsState : { ... initialState . swapsState } ,
} )
this . _fetchTradesInfo = fetchTradesInfo
this . _fetchSwapsFeatureLiveness = fetchSwapsFeatureLiveness
this . getBufferedGasLimit = getBufferedGasLimit
this . tokenRatesStore = tokenRatesStore
this . pollCount = 0
this . getProviderConfig = getProviderConfig
this . ethersProvider = new ethers . providers . Web3Provider ( provider )
this . _setupSwapsLivenessFetching ( )
}
// Once quotes are fetched, we poll for new ones to keep the quotes up to date. Market and aggregator contract conditions can change fast enough
// that quotes will no longer be available after 1 or 2 minutes. When fetchAndSetQuotes is first called it, receives fetch that parameters are stored in
// state. These stored parameters are used on subsequent calls made during polling.
// Note: we stop polling after 3 requests, until new quotes are explicitly asked for. The logic that enforces that maximum is in the body of fetchAndSetQuotes
pollForNewQuotes ( ) {
this . pollingTimeout = setTimeout ( ( ) => {
const { swapsState } = this . store . getState ( )
this . fetchAndSetQuotes ( swapsState . fetchParams , swapsState . fetchParams . metaData , true )
} , QUOTE _POLLING _INTERVAL )
}
stopPollingForQuotes ( ) {
clearTimeout ( this . pollingTimeout )
}
async fetchAndSetQuotes ( fetchParams , fetchParamsMetaData = { } , isPolledRequest ) {
if ( ! fetchParams ) {
return null
}
// Every time we get a new request that is not from the polling, we reset the poll count so we can poll for up to three more sets of quotes with these new params.
if ( ! isPolledRequest ) {
this . pollCount = 0
}
// If there are any pending poll requests, clear them so that they don't get call while this new fetch is in process
clearTimeout ( this . pollingTimeout )
if ( ! isPolledRequest ) {
this . setSwapsErrorKey ( '' )
}
let newQuotes = await this . _fetchTradesInfo ( fetchParams )
newQuotes = mapValues ( newQuotes , ( quote ) => ( {
... quote ,
sourceTokenInfo : fetchParamsMetaData . sourceTokenInfo ,
destinationTokenInfo : fetchParamsMetaData . destinationTokenInfo ,
} ) )
const quotesLastFetched = Date . now ( )
let approvalRequired = false
if ( fetchParams . sourceToken !== ETH _SWAPS _TOKEN _ADDRESS && Object . values ( newQuotes ) . length ) {
const allowance = await this . _getERC20Allowance (
fetchParams . sourceToken ,
fetchParams . fromAddress ,
)
// For a user to be able to swap a token, they need to have approved the MetaSwap contract to withdraw that token.
// _getERC20Allowance() returns the amount of the token they have approved for withdrawal. If that amount is greater
// than 0, it means that approval has already occured and is not needed. Otherwise, for tokens to be swapped, a new
// call of the ERC-20 approve method is required.
approvalRequired = allowance . eq ( 0 )
if ( ! approvalRequired ) {
newQuotes = mapValues ( newQuotes , ( quote ) => ( {
... quote ,
approvalNeeded : null ,
} ) )
} else if ( ! isPolledRequest ) {
const { gasLimit : approvalGas } = await this . timedoutGasReturn ( {
... Object . values ( newQuotes ) [ 0 ] . approvalNeeded ,
gas : DEFAULT _ERC20 _APPROVE _GAS ,
} )
newQuotes = mapValues ( newQuotes , ( quote ) => ( {
... quote ,
approvalNeeded : {
... quote . approvalNeeded ,
gas : approvalGas || DEFAULT _ERC20 _APPROVE _GAS ,
} ,
} ) )
}
}
const topAggIdData = await this . _findTopQuoteAggId ( newQuotes )
let { topAggId } = topAggIdData
const { isBest } = topAggIdData
if ( isBest ) {
newQuotes [ topAggId ] . isBestQuote = true
}
// We can reduce time on the loading screen by only doing this after the
// loading screen and best quote have rendered.
if ( ! approvalRequired && ! fetchParams ? . balanceError ) {
newQuotes = await this . getAllQuotesWithGasEstimates ( newQuotes )
if ( fetchParamsMetaData . maxMode && fetchParams . sourceToken === ETH _SWAPS _TOKEN _ADDRESS ) {
newQuotes = await this . _modifyAndFilterValuesForMaxEthMode ( newQuotes , fetchParamsMetaData . accountBalance )
}
if ( Object . values ( newQuotes ) . length === 0 ) {
this . setSwapsErrorKey ( QUOTES _NOT _AVAILABLE _ERROR )
} else {
const {
topAggId : topAggIdAfterGasEstimates ,
isBest : isBestAfterGasEstimates ,
} = await this . _findTopQuoteAggId ( newQuotes )
topAggId = topAggIdAfterGasEstimates
if ( isBestAfterGasEstimates ) {
newQuotes = mapValues ( newQuotes , ( quote ) => ( { ... quote , isBestQuote : false } ) )
newQuotes [ topAggId ] . isBestQuote = true
}
}
}
const { swapsState } = this . store . getState ( )
let { selectedAggId } = swapsState
if ( ! newQuotes [ selectedAggId ] ) {
selectedAggId = null
}
this . store . updateState ( {
swapsState : {
... swapsState ,
quotes : newQuotes ,
fetchParams : { ... fetchParams , metaData : fetchParamsMetaData } ,
quotesLastFetched ,
selectedAggId ,
topAggId ,
} ,
} )
// We only want to do up to a maximum of three requests from polling.
this . pollCount += 1
if ( this . pollCount < POLL _COUNT _LIMIT + 1 ) {
this . pollForNewQuotes ( )
} else {
this . resetPostFetchState ( )
this . setSwapsErrorKey ( QUOTES _EXPIRED _ERROR )
return null
}
return [ newQuotes , topAggId ]
}
safeRefetchQuotes ( ) {
const { swapsState } = this . store . getState ( )
if ( ! this . pollingTimeout && swapsState . fetchParams ) {
this . fetchAndSetQuotes ( swapsState . fetchParams )
}
}
setSelectedQuoteAggId ( selectedAggId ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( { swapsState : { ... swapsState , selectedAggId } } )
}
setSwapsTokens ( tokens ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( { swapsState : { ... swapsState , tokens } } )
}
setSwapsErrorKey ( errorKey ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( { swapsState : { ... swapsState , errorKey } } )
}
async getAllQuotesWithGasEstimates ( quotes ) {
const quoteGasData = await Promise . all (
Object . values ( quotes ) . map ( async ( quote ) => {
const { gasLimit , simulationFails } = await this . timedoutGasReturn ( {
... quote . trade ,
gas : ` 0x ${ ( quote . maxGas || MAX _GAS _LIMIT ) . toString ( 16 ) } ` ,
} )
return [ gasLimit , simulationFails , quote . aggregator ]
} ) ,
)
const newQuotes = { }
quoteGasData . forEach ( ( [ gasLimit , simulationFails , aggId ] ) => {
if ( gasLimit && ! simulationFails ) {
const gasEstimateWithRefund = calculateGasEstimateWithRefund ( quotes [ aggId ] . maxGas , quotes [ aggId ] . estimatedRefund , gasLimit )
newQuotes [ aggId ] = {
... quotes [ aggId ] ,
gasEstimate : gasLimit ,
gasEstimateWithRefund ,
}
} else if ( quotes [ aggId ] . approvalNeeded ) {
// If gas estimation fails, but an ERC-20 approve is needed, then we do not add any estimate property to the quote object
// Such quotes will rely on the maxGas and averageGas properties from the api
newQuotes [ aggId ] = quotes [ aggId ]
}
// If gas estimation fails and no approval is needed, then we filter that quote out, so that it is not shown to the user
} )
return newQuotes
}
timedoutGasReturn ( tradeTxParams ) {
return new Promise ( ( resolve ) => {
let gasTimedOut = false
const gasTimeout = setTimeout ( ( ) => {
gasTimedOut = true
resolve ( { gasLimit : null , simulationFails : true } )
} , 5000 )
this . getBufferedGasLimit ( { txParams : tradeTxParams } , 1 )
. then ( ( { gasLimit , simulationFails } ) => {
if ( ! gasTimedOut ) {
clearTimeout ( gasTimeout )
resolve ( { gasLimit , simulationFails } )
}
} )
. catch ( ( e ) => {
log . error ( e )
if ( ! gasTimedOut ) {
clearTimeout ( gasTimeout )
resolve ( { gasLimit : null , simulationFails : true } )
}
} )
} )
}
async setInitialGasEstimate ( initialAggId , baseGasEstimate ) {
const { swapsState } = this . store . getState ( )
const quoteToUpdate = { ... swapsState . quotes [ initialAggId ] }
const {
gasLimit : newGasEstimate ,
simulationFails ,
} = await this . timedoutGasReturn ( {
... quoteToUpdate . trade ,
gas : baseGasEstimate ,
} )
if ( newGasEstimate && ! simulationFails ) {
const gasEstimateWithRefund = calculateGasEstimateWithRefund ( quoteToUpdate . maxGas , quoteToUpdate . estimatedRefund , newGasEstimate )
quoteToUpdate . gasEstimate = newGasEstimate
quoteToUpdate . gasEstimateWithRefund = gasEstimateWithRefund
}
this . store . updateState ( {
swapsState : { ... swapsState , quotes : { ... swapsState . quotes , [ initialAggId ] : quoteToUpdate } } ,
} )
}
setApproveTxId ( approveTxId ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( { swapsState : { ... swapsState , approveTxId } } )
}
setTradeTxId ( tradeTxId ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( { swapsState : { ... swapsState , tradeTxId } } )
}
setMaxMode ( maxMode ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( { swapsState : { ... swapsState , maxMode } } )
}
setQuotesLastFetched ( quotesLastFetched ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( { swapsState : { ... swapsState , quotesLastFetched } } )
}
setSwapsTxGasPrice ( gasPrice ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( {
swapsState : { ... swapsState , customGasPrice : gasPrice } ,
} )
}
setSwapsTxGasLimit ( gasLimit ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( {
swapsState : { ... swapsState , customMaxGas : gasLimit } ,
} )
}
setCustomApproveTxData ( data ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( {
swapsState : { ... swapsState , customApproveTxData : data } ,
} )
}
setBackgroundSwapRouteState ( routeState ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( { swapsState : { ... swapsState , routeState } } )
}
setSwapsLiveness ( swapsFeatureIsLive ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( { swapsState : { ... swapsState , swapsFeatureIsLive } } )
}
resetPostFetchState ( ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( {
swapsState : {
... initialState . swapsState ,
tokens : swapsState . tokens ,
fetchParams : swapsState . fetchParams ,
swapsFeatureIsLive : swapsState . swapsFeatureIsLive ,
} ,
} )
clearTimeout ( this . pollingTimeout )
}
resetSwapsState ( ) {
const { swapsState } = this . store . getState ( )
this . store . updateState ( {
swapsState : { ... initialState . swapsState , tokens : swapsState . tokens , swapsFeatureIsLive : swapsState . swapsFeatureIsLive } ,
} )
clearTimeout ( this . pollingTimeout )
}
async _getEthersGasPrice ( ) {
const ethersGasPrice = await this . ethersProvider . getGasPrice ( )
return ethersGasPrice . toHexString ( )
}
async _findTopQuoteAggId ( quotes ) {
const tokenConversionRates = this . tokenRatesStore . getState ( )
. contractExchangeRates
const {
swapsState : { customGasPrice } ,
} = this . store . getState ( )
if ( ! Object . values ( quotes ) . length ) {
return { }
}
const usedGasPrice = customGasPrice || await this . _getEthersGasPrice ( )
let topAggId = ''
let ethValueOfTradeForBestQuote = null
Object . values ( quotes ) . forEach ( ( quote ) => {
const {
destinationAmount = 0 ,
destinationToken ,
destinationTokenInfo ,
trade ,
approvalNeeded ,
averageGas ,
gasEstimate ,
aggregator ,
} = quote
const tradeGasLimitForCalculation = gasEstimate
? new BigNumber ( gasEstimate , 16 )
: new BigNumber ( averageGas || MAX _GAS _LIMIT , 10 )
const totalGasLimitForCalculation = tradeGasLimitForCalculation
. plus ( approvalNeeded ? . gas || '0x0' , 16 )
. toString ( 16 )
const gasTotalInWeiHex = calcGasTotal (
totalGasLimitForCalculation ,
usedGasPrice ,
)
const totalEthCost = new BigNumber ( gasTotalInWeiHex , 16 ) . plus (
trade . value ,
16 ,
)
const ethFee = conversionUtil ( totalEthCost , {
fromCurrency : 'ETH' ,
fromDenomination : 'WEI' ,
toDenomination : 'ETH' ,
fromNumericBase : 'BN' ,
numberOfDecimals : 6 ,
} )
const tokenConversionRate = tokenConversionRates [ destinationToken ]
const ethValueOfTrade =
destinationTokenInfo . symbol === 'ETH'
? calcTokenAmount ( destinationAmount , 18 ) . minus ( ethFee , 10 )
: new BigNumber ( tokenConversionRate || 1 , 10 )
. times (
calcTokenAmount (
destinationAmount ,
destinationTokenInfo . decimals ,
) ,
10 ,
)
. minus ( tokenConversionRate ? ethFee . toString ( 10 ) : 0 , 10 )
if (
ethValueOfTradeForBestQuote === null ||
ethValueOfTrade . gt ( ethValueOfTradeForBestQuote )
) {
topAggId = aggregator
ethValueOfTradeForBestQuote = ethValueOfTrade
}
} )
const isBest =
quotes [ topAggId ] ? . destinationTokenInfo ? . symbol === 'ETH' ||
Boolean ( tokenConversionRates [ quotes [ topAggId ] ? . destinationToken ] )
return { topAggId , isBest }
}
async _getERC20Allowance ( contractAddress , walletAddress ) {
const contract = new ethers . Contract (
contractAddress , abi , this . ethersProvider ,
)
return await contract . allowance ( walletAddress , METASWAP _ADDRESS )
}
/ * *
* Sets up the fetching of the swaps feature liveness flag from our API .
* Performs an initial fetch when called , then fetches on a 10 - minute
* interval .
*
* If the browser goes offline , the interval is cleared and swaps are disabled
* until the value can be fetched again .
* /
_setupSwapsLivenessFetching ( ) {
const TEN _MINUTES _MS = 10 * 60 * 1000
let intervalId = null
const fetchAndSetupInterval = ( ) => {
if ( window . navigator . onLine && intervalId === null ) {
// Set the interval first to prevent race condition between listener and
// initial call to this function.
intervalId = setInterval ( this . _fetchAndSetSwapsLiveness , TEN _MINUTES _MS )
this . _fetchAndSetSwapsLiveness ( )
}
}
window . addEventListener ( 'online' , fetchAndSetupInterval )
window . addEventListener ( 'offline' , ( ) => {
if ( intervalId !== null ) {
clearInterval ( intervalId )
intervalId = null
const { swapsState } = this . store . getState ( )
if ( swapsState . swapsFeatureIsLive ) {
this . setSwapsLiveness ( false )
}
}
} )
fetchAndSetupInterval ( )
}
/ * *
* This function should only be called via _setupSwapsLivenessFetching .
*
* Attempts to fetch the swaps feature liveness flag from our API . Tries
* to fetch three times at 5 - second intervals before giving up , in which
* case the value defaults to 'false' .
*
* Only updates state if the fetched / computed flag value differs from current
* state .
* /
async _fetchAndSetSwapsLiveness ( ) {
const { swapsState } = this . store . getState ( )
const { swapsFeatureIsLive : oldSwapsFeatureIsLive } = swapsState
let swapsFeatureIsLive = false
let successfullyFetched = false
let numAttempts = 0
const fetchAndIncrementNumAttempts = async ( ) => {
try {
swapsFeatureIsLive = Boolean ( await this . _fetchSwapsFeatureLiveness ( ) )
successfullyFetched = true
} catch ( err ) {
log . error ( err )
numAttempts += 1
}
}
await fetchAndIncrementNumAttempts ( )
// The loop conditions are modified by fetchAndIncrementNumAttempts.
// eslint-disable-next-line no-unmodified-loop-condition
while ( ! successfullyFetched && numAttempts < 3 ) {
await new Promise ( ( resolve ) => {
setTimeout ( resolve , 5000 ) // 5 seconds
} )
await fetchAndIncrementNumAttempts ( )
}
if ( ! successfullyFetched ) {
log . error ( 'Failed to fetch swaps feature flag 3 times. Setting to false and trying again next interval.' )
}
if ( swapsFeatureIsLive !== oldSwapsFeatureIsLive ) {
this . setSwapsLiveness ( swapsFeatureIsLive )
}
}
async _modifyAndFilterValuesForMaxEthMode ( newQuotes , accountBalance ) {
const {
swapsState : { customGasPrice } ,
} = this . store . getState ( )
const usedGasPrice = customGasPrice || await this . _getEthersGasPrice ( )
const mappedNewQuotes = { }
Object . values ( newQuotes ) . forEach ( ( quote ) => {
const oldSourceAmount = quote . sourceAmount
const gasTotalInWeiHex = calcGasTotal ( ( new BigNumber ( quote . maxGas , 10 ) ) . toString ( 16 ) , usedGasPrice )
const newSourceAmount = ( new BigNumber ( accountBalance , 16 ) ) . minus ( gasTotalInWeiHex , 16 ) . toString ( 10 )
const newOldRatio = ( new BigNumber ( newSourceAmount , 10 ) ) . div ( oldSourceAmount , 10 )
const oldNewDifference = ( new BigNumber ( oldSourceAmount , 10 ) ) . minus ( newSourceAmount , 10 ) . toString ( 10 )
const oldDestinationAmount = quote . destinationAmount
const newDestinationAmount = ( new BigNumber ( oldDestinationAmount , 10 ) ) . times ( newOldRatio ) . toString ( 10 )
const oldValue = quote . trade . value
const newValue = ( new BigNumber ( oldValue , 16 ) ) . minus ( oldNewDifference , 10 ) . toString ( 16 )
if ( ( new BigNumber ( newSourceAmount , 10 ) . gt ( 0 ) ) ) {
mappedNewQuotes [ quote . aggregator ] = {
... quote ,
trade : {
... quote . trade ,
value : ethUtil . addHexPrefix ( newValue ) ,
} ,
destinationAmount : newDestinationAmount ,
sourceAmount : newSourceAmount ,
}
}
} )
return mappedNewQuotes
}
}