1
0
mirror of https://github.com/kremalicious/metamask-extension.git synced 2024-11-30 08:09:15 +01:00
metamask-extension/app/scripts/controllers/detect-tokens.js

185 lines
4.5 KiB
JavaScript
Raw Normal View History

import Web3 from 'web3'
import contracts from 'eth-contract-metadata'
import { warn } from 'loglevel'
import SINGLE_CALL_BALANCES_ABI from 'single-call-balance-checker-abi'
import { MAINNET } from './network/enums'
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-11-03 00:41:28 +01:00
const SINGLE_CALL_BALANCES_ADDRESS =
'0xb1f8e55c7f64d203c1400b9d8555d050f94adf39'
2018-06-27 22:29:24 +02:00
/**
* A controller that polls for token exchange
* rates based on a user's current token list
*/
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
}
/**
2020-07-20 19:02:49 +02:00
* For each token in eth-contract-metadata, find check selectedAddress balance.
2018-06-27 22:29:24 +02:00
*/
2020-11-03 00:41:28 +01:00
async detectNewTokens() {
if (!this.isActive) {
return
}
if (this._network.store.getState().provider.type !== MAINNET) {
return
}
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 &&
!this.tokenAddresses.includes(contractAddress.toLowerCase())
) {
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
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,
)
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)
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
}
return resolve(result)
})
2018-07-13 02:43:43 +02:00
})
2018-06-27 22:29:24 +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() {
if (!(this.isActive && this.selectedAddress)) {
return
}
2018-07-21 01:58:03 +02:00
this.detectNewTokens()
this.interval = DEFAULT_INTERVAL
}
/* 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)
if (!interval) {
return
}
this._handle = setInterval(() => {
this.detectNewTokens()
}, interval)
2018-06-27 22:29:24 +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) {
if (!preferences) {
return
}
2018-06-27 22:29:24 +02:00
this._preferences = preferences
const currentTokens = preferences.store.getState().tokens
this.tokenAddresses = currentTokens
? currentTokens.map((token) => token.address)
: []
preferences.store.subscribe(({ tokens = [] }) => {
this.tokenAddresses = tokens.map((token) => {
return token.address
})
})
preferences.store.subscribe(({ selectedAddress }) => {
if (this.selectedAddress !== selectedAddress) {
this.selectedAddress = selectedAddress
this.restartTokenDetection()
}
})
2018-06-27 22:29:24 +02:00
}
/**
2018-06-27 22:29:24 +02:00
* @type {Object}
*/
2020-11-03 00:41:28 +01:00
set network(network) {
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-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) {
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
if (isUnlocked) {
this.restartTokenDetection()
}
}
2018-07-20 01:46:46 +02:00
})
}
2018-07-21 22:03:31 +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
}
/* eslint-enable accessor-pairs */
2018-06-27 22:29:24 +02:00
}