You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
323 lines
11 KiB
323 lines
11 KiB
import { ObservableStore } from '@metamask/obs-store';
|
|
import log from 'loglevel';
|
|
import BN from 'bn.js';
|
|
import createId from '../../../shared/modules/random-id';
|
|
import { bnToHex } from '../lib/util';
|
|
import getFetchWithTimeout from '../../../shared/modules/fetch-with-timeout';
|
|
|
|
import {
|
|
TRANSACTION_TYPES,
|
|
TRANSACTION_STATUSES,
|
|
} from '../../../shared/constants/transaction';
|
|
import {
|
|
CHAIN_IDS,
|
|
CHAIN_ID_TO_NETWORK_ID_MAP,
|
|
CHAIN_ID_TO_TYPE_MAP,
|
|
} from '../../../shared/constants/network';
|
|
|
|
const fetchWithTimeout = getFetchWithTimeout();
|
|
|
|
/**
|
|
* @typedef {import('../../../shared/constants/transaction').TransactionMeta} TransactionMeta
|
|
*/
|
|
|
|
/**
|
|
* A transaction object in the format returned by the Etherscan API.
|
|
*
|
|
* Note that this is not an exhaustive type definiton; only the properties we use are defined
|
|
*
|
|
* @typedef {object} EtherscanTransaction
|
|
* @property {string} blockNumber - The number of the block this transaction was found in, in decimal
|
|
* @property {string} from - The hex-prefixed address of the sender
|
|
* @property {string} gas - The gas limit, in decimal GWEI
|
|
* @property {string} [gasPrice] - The gas price, in decimal WEI
|
|
* @property {string} [maxFeePerGas] - The maximum fee per gas, inclusive of tip, in decimal WEI
|
|
* @property {string} [maxPriorityFeePerGas] - The maximum tip per gas in decimal WEI
|
|
* @property {string} hash - The hex-prefixed transaction hash
|
|
* @property {string} isError - Whether the transaction was confirmed or failed (0 for confirmed, 1 for failed)
|
|
* @property {string} nonce - The transaction nonce, in decimal
|
|
* @property {string} timeStamp - The timestamp for the transaction, in seconds
|
|
* @property {string} to - The hex-prefixed address of the recipient
|
|
* @property {string} value - The amount of ETH sent in this transaction, in decimal WEI
|
|
*/
|
|
|
|
/**
|
|
* This controller is responsible for retrieving incoming transactions. Etherscan is polled once every block to check
|
|
* for new incoming transactions for the current selected account on the current network
|
|
*
|
|
* Note that only the built-in Infura networks are supported (i.e. anything in `INFURA_PROVIDER_TYPES`). We will not
|
|
* attempt to retrieve incoming transactions on any custom RPC endpoints.
|
|
*/
|
|
const etherscanSupportedNetworks = [
|
|
CHAIN_IDS.GOERLI,
|
|
CHAIN_IDS.MAINNET,
|
|
CHAIN_IDS.SEPOLIA,
|
|
];
|
|
|
|
export default class IncomingTransactionsController {
|
|
constructor(opts = {}) {
|
|
const {
|
|
blockTracker,
|
|
onNetworkDidChange,
|
|
getCurrentChainId,
|
|
preferencesController,
|
|
} = opts;
|
|
this.blockTracker = blockTracker;
|
|
this.getCurrentChainId = getCurrentChainId;
|
|
this.preferencesController = preferencesController;
|
|
|
|
this._onLatestBlock = async (newBlockNumberHex) => {
|
|
const selectedAddress = this.preferencesController.getSelectedAddress();
|
|
const newBlockNumberDec = parseInt(newBlockNumberHex, 16);
|
|
await this._update(selectedAddress, newBlockNumberDec);
|
|
};
|
|
|
|
const initState = {
|
|
incomingTransactions: {},
|
|
incomingTxLastFetchedBlockByChainId: {
|
|
[CHAIN_IDS.GOERLI]: null,
|
|
[CHAIN_IDS.MAINNET]: null,
|
|
[CHAIN_IDS.SEPOLIA]: null,
|
|
},
|
|
...opts.initState,
|
|
};
|
|
this.store = new ObservableStore(initState);
|
|
|
|
this.preferencesController.store.subscribe(
|
|
previousValueComparator((prevState, currState) => {
|
|
const {
|
|
featureFlags: {
|
|
showIncomingTransactions: prevShowIncomingTransactions,
|
|
} = {},
|
|
} = prevState;
|
|
const {
|
|
featureFlags: {
|
|
showIncomingTransactions: currShowIncomingTransactions,
|
|
} = {},
|
|
} = currState;
|
|
|
|
if (currShowIncomingTransactions === prevShowIncomingTransactions) {
|
|
return;
|
|
}
|
|
|
|
if (prevShowIncomingTransactions && !currShowIncomingTransactions) {
|
|
this.stop();
|
|
return;
|
|
}
|
|
|
|
this.start();
|
|
}, this.preferencesController.store.getState()),
|
|
);
|
|
|
|
this.preferencesController.store.subscribe(
|
|
previousValueComparator(async (prevState, currState) => {
|
|
const { selectedAddress: prevSelectedAddress } = prevState;
|
|
const { selectedAddress: currSelectedAddress } = currState;
|
|
|
|
if (currSelectedAddress === prevSelectedAddress) {
|
|
return;
|
|
}
|
|
await this._update(currSelectedAddress);
|
|
}, this.preferencesController.store.getState()),
|
|
);
|
|
|
|
onNetworkDidChange(async () => {
|
|
const address = this.preferencesController.getSelectedAddress();
|
|
await this._update(address);
|
|
});
|
|
}
|
|
|
|
start() {
|
|
const { featureFlags = {} } = this.preferencesController.store.getState();
|
|
const { showIncomingTransactions } = featureFlags;
|
|
|
|
if (!showIncomingTransactions) {
|
|
return;
|
|
}
|
|
|
|
this.blockTracker.removeListener('latest', this._onLatestBlock);
|
|
this.blockTracker.addListener('latest', this._onLatestBlock);
|
|
}
|
|
|
|
stop() {
|
|
this.blockTracker.removeListener('latest', this._onLatestBlock);
|
|
}
|
|
|
|
/**
|
|
* Determines the correct block number to begin looking for new transactions
|
|
* from, fetches the transactions and then saves them and the next block
|
|
* number to begin fetching from in state. Block numbers and transactions are
|
|
* stored per chainId.
|
|
*
|
|
* @private
|
|
* @param {string} address - address to lookup transactions for
|
|
* @param {number} [newBlockNumberDec] - block number to begin fetching from
|
|
*/
|
|
async _update(address, newBlockNumberDec) {
|
|
const chainId = this.getCurrentChainId();
|
|
if (!etherscanSupportedNetworks.includes(chainId) || !address) {
|
|
return;
|
|
}
|
|
try {
|
|
const currentState = this.store.getState();
|
|
const currentBlock = parseInt(this.blockTracker.getCurrentBlock(), 16);
|
|
|
|
const mostRecentlyFetchedBlock =
|
|
currentState.incomingTxLastFetchedBlockByChainId[chainId];
|
|
const blockToFetchFrom =
|
|
mostRecentlyFetchedBlock ?? newBlockNumberDec ?? currentBlock;
|
|
|
|
const newIncomingTxs = await this._getNewIncomingTransactions(
|
|
address,
|
|
blockToFetchFrom,
|
|
chainId,
|
|
);
|
|
|
|
let newMostRecentlyFetchedBlock = blockToFetchFrom;
|
|
|
|
newIncomingTxs.forEach((tx) => {
|
|
if (
|
|
tx.blockNumber &&
|
|
parseInt(newMostRecentlyFetchedBlock, 10) <
|
|
parseInt(tx.blockNumber, 10)
|
|
) {
|
|
newMostRecentlyFetchedBlock = parseInt(tx.blockNumber, 10);
|
|
}
|
|
});
|
|
|
|
this.store.updateState({
|
|
incomingTxLastFetchedBlockByChainId: {
|
|
...currentState.incomingTxLastFetchedBlockByChainId,
|
|
[chainId]: newMostRecentlyFetchedBlock + 1,
|
|
},
|
|
incomingTransactions: newIncomingTxs.reduce(
|
|
(transactions, tx) => {
|
|
transactions[tx.hash] = tx;
|
|
return transactions;
|
|
},
|
|
{
|
|
...currentState.incomingTransactions,
|
|
},
|
|
),
|
|
});
|
|
} catch (err) {
|
|
log.error(err);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* fetches transactions for the given address and chain, via etherscan, then
|
|
* processes the data into the necessary shape for usage in this controller.
|
|
*
|
|
* @private
|
|
* @param {string} [address] - Address to fetch transactions for
|
|
* @param {number} [fromBlock] - Block to look for transactions at
|
|
* @param {string} [chainId] - The chainId for the current network
|
|
* @returns {TransactionMeta[]}
|
|
*/
|
|
async _getNewIncomingTransactions(address, fromBlock, chainId) {
|
|
const etherscanSubdomain =
|
|
chainId === CHAIN_IDS.MAINNET
|
|
? 'api'
|
|
: `api-${CHAIN_ID_TO_TYPE_MAP[chainId]}`;
|
|
|
|
const apiUrl = `https://${etherscanSubdomain}.etherscan.io`;
|
|
let url = `${apiUrl}/api?module=account&action=txlist&address=${address}&tag=latest&page=1`;
|
|
|
|
if (fromBlock) {
|
|
url += `&startBlock=${parseInt(fromBlock, 10)}`;
|
|
}
|
|
const response = await fetchWithTimeout(url);
|
|
const { status, result } = await response.json();
|
|
let newIncomingTxs = [];
|
|
if (status === '1' && Array.isArray(result) && result.length > 0) {
|
|
const remoteTxList = {};
|
|
const remoteTxs = [];
|
|
result.forEach((tx) => {
|
|
if (!remoteTxList[tx.hash]) {
|
|
remoteTxs.push(this._normalizeTxFromEtherscan(tx, chainId));
|
|
remoteTxList[tx.hash] = 1;
|
|
}
|
|
});
|
|
|
|
newIncomingTxs = remoteTxs.filter(
|
|
(tx) => tx.txParams?.to?.toLowerCase() === address.toLowerCase(),
|
|
);
|
|
newIncomingTxs.sort((a, b) => (a.time < b.time ? -1 : 1));
|
|
}
|
|
return newIncomingTxs;
|
|
}
|
|
|
|
/**
|
|
* Transmutes a EtherscanTransaction into a TransactionMeta
|
|
*
|
|
* @param {EtherscanTransaction} etherscanTransaction - the transaction to normalize
|
|
* @param {string} chainId - The chainId of the current network
|
|
* @returns {TransactionMeta}
|
|
*/
|
|
_normalizeTxFromEtherscan(etherscanTransaction, chainId) {
|
|
const time = parseInt(etherscanTransaction.timeStamp, 10) * 1000;
|
|
const status =
|
|
etherscanTransaction.isError === '0'
|
|
? TRANSACTION_STATUSES.CONFIRMED
|
|
: TRANSACTION_STATUSES.FAILED;
|
|
const txParams = {
|
|
from: etherscanTransaction.from,
|
|
gas: bnToHex(new BN(etherscanTransaction.gas)),
|
|
nonce: bnToHex(new BN(etherscanTransaction.nonce)),
|
|
to: etherscanTransaction.to,
|
|
value: bnToHex(new BN(etherscanTransaction.value)),
|
|
};
|
|
|
|
if (etherscanTransaction.gasPrice) {
|
|
txParams.gasPrice = bnToHex(new BN(etherscanTransaction.gasPrice));
|
|
} else if (etherscanTransaction.maxFeePerGas) {
|
|
txParams.maxFeePerGas = bnToHex(
|
|
new BN(etherscanTransaction.maxFeePerGas),
|
|
);
|
|
txParams.maxPriorityFeePerGas = bnToHex(
|
|
new BN(etherscanTransaction.maxPriorityFeePerGas),
|
|
);
|
|
}
|
|
|
|
return {
|
|
blockNumber: etherscanTransaction.blockNumber,
|
|
id: createId(),
|
|
chainId,
|
|
metamaskNetworkId: CHAIN_ID_TO_NETWORK_ID_MAP[chainId],
|
|
status,
|
|
time,
|
|
txParams,
|
|
hash: etherscanTransaction.hash,
|
|
type: TRANSACTION_TYPES.INCOMING,
|
|
};
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Returns a function with arity 1 that caches the argument that the function
|
|
* is called with and invokes the comparator with both the cached, previous,
|
|
* value and the current value. If specified, the initialValue will be passed
|
|
* in as the previous value on the first invocation of the returned method.
|
|
*
|
|
* @template A - The type of the compared value.
|
|
* @param {(prevValue: A, nextValue: A) => void} comparator - A method to compare
|
|
* the previous and next values.
|
|
* @param {A} [initialValue] - The initial value to supply to prevValue
|
|
* on first call of the method.
|
|
*/
|
|
function previousValueComparator(comparator, initialValue) {
|
|
let first = true;
|
|
let cache;
|
|
return (value) => {
|
|
try {
|
|
if (first) {
|
|
first = false;
|
|
return comparator(initialValue ?? value, value);
|
|
}
|
|
return comparator(cache, value);
|
|
} finally {
|
|
cache = value;
|
|
}
|
|
};
|
|
}
|
|
|