2020-01-09 04:34:58 +01:00
|
|
|
import Web3 from 'web3'
|
2020-12-01 23:55:01 +01:00
|
|
|
import contracts from '@metamask/contract-metadata'
|
2020-01-09 04:34:58 +01:00
|
|
|
import { warn } from 'loglevel'
|
2020-08-18 21:18:25 +02:00
|
|
|
import SINGLE_CALL_BALANCES_ABI from 'single-call-balance-checker-abi'
|
2021-01-20 17:13:14 +01:00
|
|
|
import { MAINNET_CHAIN_ID } from '../../../shared/constants/network'
|
|
|
|
import { SINGLE_CALL_BALANCES_ADDRESS } from '../constants/contracts'
|
2020-08-18 21:18:25 +02:00
|
|
|
|
2018-06-27 22:29:24 +02:00
|
|
|
// By default, poll every 3 minutes
|
2018-06-28 04:18:06 +02:00
|
|
|
const DEFAULT_INTERVAL = 180 * 1000
|
2020-08-19 18:27:05 +02:00
|
|
|
|
2018-06-27 22:29:24 +02:00
|
|
|
/**
|
|
|
|
* A controller that polls for token exchange
|
|
|
|
* rates based on a user's current token list
|
|
|
|
*/
|
2020-05-06 00:19:38 +02:00
|
|
|
export default class DetectTokensController {
|
2018-06-27 22:29:24 +02:00
|
|
|
/**
|
|
|
|
* Creates a DetectTokensController
|
|
|
|
*
|
|
|
|
* @param {Object} [config] - Options to configure controller
|
|
|
|
*/
|
2020-11-03 00:41:28 +01:00
|
|
|
constructor({
|
|
|
|
interval = DEFAULT_INTERVAL,
|
|
|
|
preferences,
|
|
|
|
network,
|
|
|
|
keyringMemStore,
|
|
|
|
} = {}) {
|
2018-06-27 22:29:24 +02:00
|
|
|
this.preferences = preferences
|
|
|
|
this.interval = interval
|
|
|
|
this.network = network
|
2018-07-20 01:46:46 +02:00
|
|
|
this.keyringMemStore = keyringMemStore
|
2018-06-27 22:29:24 +02:00
|
|
|
}
|
|
|
|
|
2018-07-19 21:56:38 +02:00
|
|
|
/**
|
2020-12-01 23:55:01 +01:00
|
|
|
* For each token in @metamask/contract-metadata, find check selectedAddress balance.
|
2018-06-27 22:29:24 +02:00
|
|
|
*/
|
2020-11-03 00:41:28 +01:00
|
|
|
async detectNewTokens() {
|
2019-11-20 01:03:20 +01:00
|
|
|
if (!this.isActive) {
|
|
|
|
return
|
|
|
|
}
|
2021-01-08 15:25:46 +01:00
|
|
|
if (this._network.store.getState().provider.chainId !== MAINNET_CHAIN_ID) {
|
2019-11-20 01:03:20 +01:00
|
|
|
return
|
|
|
|
}
|
2020-04-13 22:14:42 +02:00
|
|
|
|
2019-01-22 19:23:11 +01:00
|
|
|
const tokensToDetect = []
|
2018-07-13 02:43:43 +02:00
|
|
|
this.web3.setProvider(this._network._provider)
|
2018-07-11 21:59:05 +02:00
|
|
|
for (const contractAddress in contracts) {
|
2020-11-03 00:41:28 +01:00
|
|
|
if (
|
|
|
|
contracts[contractAddress].erc20 &&
|
2020-12-08 21:38:00 +01:00
|
|
|
!this.tokenAddresses.includes(contractAddress.toLowerCase()) &&
|
|
|
|
!this.hiddenTokens.includes(contractAddress.toLowerCase())
|
2020-11-03 00:41:28 +01:00
|
|
|
) {
|
2019-01-22 19:23:11 +01:00
|
|
|
tokensToDetect.push(contractAddress)
|
2018-07-13 02:43:43 +02:00
|
|
|
}
|
2018-06-27 22:29:24 +02:00
|
|
|
}
|
2019-01-22 19:23:11 +01:00
|
|
|
|
2020-04-13 22:14:42 +02:00
|
|
|
let result
|
|
|
|
try {
|
|
|
|
result = await this._getTokenBalances(tokensToDetect)
|
|
|
|
} catch (error) {
|
2020-11-03 00:41:28 +01:00
|
|
|
warn(
|
|
|
|
`MetaMask - DetectTokensController single call balance fetch failed`,
|
|
|
|
error,
|
|
|
|
)
|
2020-04-13 22:14:42 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
tokensToDetect.forEach((tokenAddress, index) => {
|
|
|
|
const balance = result[index]
|
|
|
|
if (balance && !balance.isZero()) {
|
2020-11-03 00:41:28 +01:00
|
|
|
this._preferences.addToken(
|
|
|
|
tokenAddress,
|
|
|
|
contracts[tokenAddress].symbol,
|
|
|
|
contracts[tokenAddress].decimals,
|
|
|
|
)
|
2019-01-22 19:23:11 +01:00
|
|
|
}
|
|
|
|
})
|
2018-06-27 22:29:24 +02:00
|
|
|
}
|
|
|
|
|
2020-11-03 00:41:28 +01:00
|
|
|
async _getTokenBalances(tokens) {
|
|
|
|
const ethContract = this.web3.eth
|
|
|
|
.contract(SINGLE_CALL_BALANCES_ABI)
|
|
|
|
.at(SINGLE_CALL_BALANCES_ADDRESS)
|
2020-04-13 22:14:42 +02:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
ethContract.balances([this.selectedAddress], tokens, (error, result) => {
|
|
|
|
if (error) {
|
|
|
|
return reject(error)
|
2018-07-13 02:43:43 +02:00
|
|
|
}
|
2020-04-13 22:14:42 +02:00
|
|
|
return resolve(result)
|
|
|
|
})
|
2018-07-13 02:43:43 +02:00
|
|
|
})
|
2018-06-27 22:29:24 +02:00
|
|
|
}
|
|
|
|
|
2018-07-19 21:56:38 +02:00
|
|
|
/**
|
|
|
|
* Restart token detection polling period and call detectNewTokens
|
|
|
|
* in case of address change or user session initialization.
|
|
|
|
*
|
|
|
|
*/
|
2020-11-03 00:41:28 +01:00
|
|
|
restartTokenDetection() {
|
2019-11-20 01:03:20 +01:00
|
|
|
if (!(this.isActive && this.selectedAddress)) {
|
|
|
|
return
|
|
|
|
}
|
2018-07-21 01:58:03 +02:00
|
|
|
this.detectNewTokens()
|
|
|
|
this.interval = DEFAULT_INTERVAL
|
2018-07-19 21:56:38 +02:00
|
|
|
}
|
|
|
|
|
2020-10-22 19:06:44 +02:00
|
|
|
/* eslint-disable accessor-pairs */
|
2018-06-27 22:29:24 +02:00
|
|
|
/**
|
|
|
|
* @type {Number}
|
|
|
|
*/
|
2020-11-03 00:41:28 +01:00
|
|
|
set interval(interval) {
|
2018-06-27 22:29:24 +02:00
|
|
|
this._handle && clearInterval(this._handle)
|
2019-11-20 01:03:20 +01:00
|
|
|
if (!interval) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
this._handle = setInterval(() => {
|
|
|
|
this.detectNewTokens()
|
|
|
|
}, interval)
|
2018-06-27 22:29:24 +02:00
|
|
|
}
|
|
|
|
|
2018-07-19 21:56:38 +02:00
|
|
|
/**
|
|
|
|
* In setter when selectedAddress is changed, detectNewTokens and restart polling
|
2018-06-27 22:29:24 +02:00
|
|
|
* @type {Object}
|
|
|
|
*/
|
2020-11-03 00:41:28 +01:00
|
|
|
set preferences(preferences) {
|
2019-11-20 01:03:20 +01:00
|
|
|
if (!preferences) {
|
|
|
|
return
|
|
|
|
}
|
2018-06-27 22:29:24 +02:00
|
|
|
this._preferences = preferences
|
2020-04-13 22:14:42 +02:00
|
|
|
const currentTokens = preferences.store.getState().tokens
|
|
|
|
this.tokenAddresses = currentTokens
|
|
|
|
? currentTokens.map((token) => token.address)
|
|
|
|
: []
|
2020-12-08 21:38:00 +01:00
|
|
|
this.hiddenTokens = preferences.store.getState().hiddenTokens
|
|
|
|
preferences.store.subscribe(({ tokens = [], hiddenTokens = [] }) => {
|
2020-04-13 22:14:42 +02:00
|
|
|
this.tokenAddresses = tokens.map((token) => {
|
|
|
|
return token.address
|
2019-11-20 01:03:20 +01:00
|
|
|
})
|
2020-12-08 21:38:00 +01:00
|
|
|
this.hiddenTokens = hiddenTokens
|
2019-11-20 01:03:20 +01:00
|
|
|
})
|
2018-07-19 21:56:38 +02:00
|
|
|
preferences.store.subscribe(({ selectedAddress }) => {
|
|
|
|
if (this.selectedAddress !== selectedAddress) {
|
|
|
|
this.selectedAddress = selectedAddress
|
|
|
|
this.restartTokenDetection()
|
|
|
|
}
|
|
|
|
})
|
2018-06-27 22:29:24 +02:00
|
|
|
}
|
|
|
|
|
2018-07-19 21:56:38 +02:00
|
|
|
/**
|
2018-06-27 22:29:24 +02:00
|
|
|
* @type {Object}
|
|
|
|
*/
|
2020-11-03 00:41:28 +01:00
|
|
|
set network(network) {
|
2019-11-20 01:03:20 +01:00
|
|
|
if (!network) {
|
|
|
|
return
|
|
|
|
}
|
2018-06-27 22:29:24 +02:00
|
|
|
this._network = network
|
2018-07-13 02:43:43 +02:00
|
|
|
this.web3 = new Web3(network._provider)
|
2018-06-27 22:29:24 +02:00
|
|
|
}
|
2018-07-19 21:56:38 +02:00
|
|
|
|
|
|
|
/**
|
2018-07-20 01:46:46 +02:00
|
|
|
* In setter when isUnlocked is updated to true, detectNewTokens and restart polling
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
2020-11-03 00:41:28 +01:00
|
|
|
set keyringMemStore(keyringMemStore) {
|
2019-11-20 01:03:20 +01:00
|
|
|
if (!keyringMemStore) {
|
|
|
|
return
|
|
|
|
}
|
2018-07-20 01:46:46 +02:00
|
|
|
this._keyringMemStore = keyringMemStore
|
|
|
|
this._keyringMemStore.subscribe(({ isUnlocked }) => {
|
|
|
|
if (this.isUnlocked !== isUnlocked) {
|
|
|
|
this.isUnlocked = isUnlocked
|
2019-11-20 01:03:20 +01:00
|
|
|
if (isUnlocked) {
|
|
|
|
this.restartTokenDetection()
|
|
|
|
}
|
2018-07-19 21:56:38 +02:00
|
|
|
}
|
2018-07-20 01:46:46 +02:00
|
|
|
})
|
|
|
|
}
|
2018-07-21 22:03:31 +02:00
|
|
|
|
2018-07-21 02:09:37 +02:00
|
|
|
/**
|
|
|
|
* Internal isActive state
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
2020-11-03 00:41:28 +01:00
|
|
|
get isActive() {
|
2018-07-21 01:58:03 +02:00
|
|
|
return this.isOpen && this.isUnlocked
|
|
|
|
}
|
2020-10-22 19:06:44 +02:00
|
|
|
/* eslint-enable accessor-pairs */
|
2018-06-27 22:29:24 +02:00
|
|
|
}
|