mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-12-12 12:47:14 +01:00
6e5c2f03bf
* addding the legacy tokenlist, tuning token detection OFF by default, adding new message while importing tokens updating the controller version and calling detectNewToken on network change fixing rebase error Run yarn lavamoat:auto for updating policies updating lavamoat Deleted node modules and run again lavamoat auto fixing rebase issues updating lavamoat policies updating lavamoat after rebasing policies updating custom token warning and blocking detectedtoken link when tpken detection is off for supported networks to update the token in fetchTosync updating the contract map object Revert build-system lavamoat policy changes Move token list selection logic from components to getTokenList selector updating the tokenList Update lavamoat Fix error updating lavamoat lint fix fix unit test fail fix unit test fail lint fix fixing rebase locale error rebase fix Revert build-system policy changes temp addressing review comments * rebase fix
307 lines
9.3 KiB
JavaScript
307 lines
9.3 KiB
JavaScript
import log from 'loglevel';
|
|
import BigNumber from 'bignumber.js';
|
|
import {
|
|
conversionUtil,
|
|
multiplyCurrencies,
|
|
} from '../../../shared/modules/conversion.utils';
|
|
import { getTokenStandardAndDetails } from '../../store/actions';
|
|
import { isEqualCaseInsensitive } from '../../../shared/modules/string-utils';
|
|
import { parseStandardTokenTransactionData } from '../../../shared/modules/transaction.utils';
|
|
import { ERC20 } from '../../../shared/constants/transaction';
|
|
import * as util from './util';
|
|
import { formatCurrency } from './confirm-tx.util';
|
|
|
|
const DEFAULT_SYMBOL = '';
|
|
|
|
async function getSymbolFromContract(tokenAddress) {
|
|
const token = util.getContractAtAddress(tokenAddress);
|
|
try {
|
|
const result = await token.symbol();
|
|
return result[0];
|
|
} catch (error) {
|
|
log.warn(
|
|
`symbol() call for token at address ${tokenAddress} resulted in error:`,
|
|
error,
|
|
);
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
async function getDecimalsFromContract(tokenAddress) {
|
|
const token = util.getContractAtAddress(tokenAddress);
|
|
|
|
try {
|
|
const result = await token.decimals();
|
|
const decimalsBN = result[0];
|
|
return decimalsBN?.toString();
|
|
} catch (error) {
|
|
log.warn(
|
|
`decimals() call for token at address ${tokenAddress} resulted in error:`,
|
|
error,
|
|
);
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
export function getTokenMetadata(tokenAddress, tokenList) {
|
|
return tokenAddress && tokenList[tokenAddress.toLowerCase()];
|
|
}
|
|
|
|
async function getSymbol(tokenAddress, tokenList) {
|
|
let symbol = await getSymbolFromContract(tokenAddress);
|
|
|
|
if (!symbol) {
|
|
const contractMetadataInfo = getTokenMetadata(tokenAddress, tokenList);
|
|
|
|
if (contractMetadataInfo) {
|
|
symbol = contractMetadataInfo.symbol;
|
|
}
|
|
}
|
|
|
|
return symbol;
|
|
}
|
|
|
|
async function getDecimals(tokenAddress, tokenList) {
|
|
let decimals = await getDecimalsFromContract(tokenAddress);
|
|
|
|
if (!decimals || decimals === '0') {
|
|
const contractMetadataInfo = getTokenMetadata(tokenAddress, tokenList);
|
|
|
|
if (contractMetadataInfo) {
|
|
decimals = contractMetadataInfo.decimals?.toString();
|
|
}
|
|
}
|
|
|
|
return decimals;
|
|
}
|
|
|
|
export async function getSymbolAndDecimals(tokenAddress, tokenList) {
|
|
let symbol, decimals;
|
|
|
|
try {
|
|
symbol = await getSymbol(tokenAddress, tokenList);
|
|
decimals = await getDecimals(tokenAddress, tokenList);
|
|
} catch (error) {
|
|
log.warn(
|
|
`symbol() and decimal() calls for token at address ${tokenAddress} resulted in error:`,
|
|
error,
|
|
);
|
|
}
|
|
|
|
return {
|
|
symbol: symbol || DEFAULT_SYMBOL,
|
|
decimals,
|
|
};
|
|
}
|
|
|
|
export function tokenInfoGetter() {
|
|
const tokens = {};
|
|
|
|
return async (address, tokenList) => {
|
|
if (tokens[address]) {
|
|
return tokens[address];
|
|
}
|
|
|
|
tokens[address] = await getSymbolAndDecimals(address, tokenList);
|
|
|
|
return tokens[address];
|
|
};
|
|
}
|
|
|
|
export function calcTokenAmount(value, decimals) {
|
|
const multiplier = Math.pow(10, Number(decimals || 0));
|
|
return new BigNumber(String(value)).div(multiplier);
|
|
}
|
|
|
|
export function calcTokenValue(value, decimals) {
|
|
const multiplier = Math.pow(10, Number(decimals || 0));
|
|
return new BigNumber(String(value)).times(multiplier);
|
|
}
|
|
|
|
/**
|
|
* Attempts to get the address parameter of the given token transaction data
|
|
* (i.e. function call) per the Human Standard Token ABI, in the following
|
|
* order:
|
|
* - The '_to' parameter, if present
|
|
* - The first parameter, if present
|
|
*
|
|
* @param {object} tokenData - ethers Interface token data.
|
|
* @returns {string | undefined} A lowercase address string.
|
|
*/
|
|
export function getTokenAddressParam(tokenData = {}) {
|
|
const value =
|
|
tokenData?.args?._to || tokenData?.args?.to || tokenData?.args?.[0];
|
|
return value?.toString().toLowerCase();
|
|
}
|
|
|
|
/**
|
|
* Gets the '_value' parameter of the given token transaction data
|
|
* (i.e function call) per the Human Standard Token ABI, if present.
|
|
*
|
|
* @param {object} tokenData - ethers Interface token data.
|
|
* @returns {string | undefined} A decimal string value.
|
|
*/
|
|
export function getTokenValueParam(tokenData = {}) {
|
|
return tokenData?.args?._value?.toString();
|
|
}
|
|
|
|
/**
|
|
* Gets either the '_tokenId' parameter or the 'id' param of the passed token transaction data.,
|
|
* These are the parsed tokenId values returned by `parseStandardTokenTransactionData` as defined
|
|
* in the ERC721 and ERC1155 ABIs from metamask-eth-abis (https://github.com/MetaMask/metamask-eth-abis/tree/main/src/abis)
|
|
*
|
|
* @param {object} tokenData - ethers Interface token data.
|
|
* @returns {string | undefined} A decimal string value.
|
|
*/
|
|
export function getTokenIdParam(tokenData = {}) {
|
|
return (
|
|
tokenData?.args?._tokenId?.toString() ?? tokenData?.args?.id?.toString()
|
|
);
|
|
}
|
|
|
|
/**
|
|
* Gets the '_approved' parameter of the given token transaction data
|
|
* (i.e function call) per the Human Standard Token ABI, if present.
|
|
*
|
|
* @param {object} tokenData - ethers Interface token data.
|
|
* @returns {boolean | undefined} A boolean indicating whether the function is being called to approve or revoke access.
|
|
*/
|
|
export function getTokenApprovedParam(tokenData = {}) {
|
|
return tokenData?.args?._approved;
|
|
}
|
|
|
|
/**
|
|
* Get the token balance converted to fiat and optionally formatted for display
|
|
*
|
|
* @param {number} [contractExchangeRate] - The exchange rate between the current token and the native currency
|
|
* @param {number} conversionRate - The exchange rate between the current fiat currency and the native currency
|
|
* @param {string} currentCurrency - The currency code for the user's chosen fiat currency
|
|
* @param {string} [tokenAmount] - The current token balance
|
|
* @param {string} [tokenSymbol] - The token symbol
|
|
* @param {boolean} [formatted] - Whether the return value should be formatted or not
|
|
* @param {boolean} [hideCurrencySymbol] - excludes the currency symbol in the result if true
|
|
* @returns {string|undefined} The token amount in the user's chosen fiat currency, optionally formatted and localize
|
|
*/
|
|
export function getTokenFiatAmount(
|
|
contractExchangeRate,
|
|
conversionRate,
|
|
currentCurrency,
|
|
tokenAmount,
|
|
tokenSymbol,
|
|
formatted = true,
|
|
hideCurrencySymbol = false,
|
|
) {
|
|
// If the conversionRate is 0 (i.e. unknown) or the contract exchange rate
|
|
// is currently unknown, the fiat amount cannot be calculated so it is not
|
|
// shown to the user
|
|
if (
|
|
conversionRate <= 0 ||
|
|
!contractExchangeRate ||
|
|
tokenAmount === undefined
|
|
) {
|
|
return undefined;
|
|
}
|
|
|
|
const currentTokenToFiatRate = multiplyCurrencies(
|
|
contractExchangeRate,
|
|
conversionRate,
|
|
{
|
|
multiplicandBase: 10,
|
|
multiplierBase: 10,
|
|
},
|
|
);
|
|
const currentTokenInFiat = conversionUtil(tokenAmount, {
|
|
fromNumericBase: 'dec',
|
|
fromCurrency: tokenSymbol,
|
|
toCurrency: currentCurrency.toUpperCase(),
|
|
numberOfDecimals: 2,
|
|
conversionRate: currentTokenToFiatRate,
|
|
});
|
|
let result;
|
|
if (hideCurrencySymbol) {
|
|
result = formatCurrency(currentTokenInFiat, currentCurrency);
|
|
} else if (formatted) {
|
|
result = `${formatCurrency(
|
|
currentTokenInFiat,
|
|
currentCurrency,
|
|
)} ${currentCurrency.toUpperCase()}`;
|
|
} else {
|
|
result = currentTokenInFiat;
|
|
}
|
|
return result;
|
|
}
|
|
|
|
export async function getAssetDetails(
|
|
tokenAddress,
|
|
currentUserAddress,
|
|
transactionData,
|
|
existingCollectibles,
|
|
) {
|
|
const tokenData = parseStandardTokenTransactionData(transactionData);
|
|
if (!tokenData) {
|
|
throw new Error('Unable to detect valid token data');
|
|
}
|
|
|
|
// Sometimes the tokenId value is parsed as "_value" param. Not seeing this often any more, but still occasionally:
|
|
// i.e. call approve() on BAYC contract - https://etherscan.io/token/0xbc4ca0eda7647a8ab7c2061c2e118a18a936f13d#writeContract, and tokenId shows up as _value,
|
|
// not sure why since it doesn't match the ERC721 ABI spec we use to parse these transactions - https://github.com/MetaMask/metamask-eth-abis/blob/d0474308a288f9252597b7c93a3a8deaad19e1b2/src/abis/abiERC721.ts#L62.
|
|
let tokenId =
|
|
getTokenIdParam(tokenData)?.toString() ?? getTokenValueParam(tokenData);
|
|
|
|
const toAddress = getTokenAddressParam(tokenData);
|
|
|
|
let tokenDetails;
|
|
|
|
// if a tokenId is present check if there is a collectible in state matching the address/tokenId
|
|
// and avoid unnecessary network requests to query token details we already have
|
|
if (existingCollectibles?.length && tokenId) {
|
|
const existingCollectible = existingCollectibles.find(
|
|
({ address, tokenId: _tokenId }) =>
|
|
isEqualCaseInsensitive(tokenAddress, address) && _tokenId === tokenId,
|
|
);
|
|
|
|
if (existingCollectible) {
|
|
return {
|
|
toAddress,
|
|
...existingCollectible,
|
|
};
|
|
}
|
|
}
|
|
|
|
try {
|
|
tokenDetails = await getTokenStandardAndDetails(
|
|
tokenAddress,
|
|
currentUserAddress,
|
|
tokenId,
|
|
);
|
|
} catch (error) {
|
|
log.warn(error);
|
|
// if we can't determine any token standard or details return the data we can extract purely from the parsed transaction data
|
|
return { toAddress, tokenId };
|
|
}
|
|
|
|
const tokenAmount =
|
|
tokenData &&
|
|
tokenDetails?.decimals &&
|
|
calcTokenAmount(
|
|
getTokenValueParam(tokenData),
|
|
tokenDetails?.decimals,
|
|
).toString(10);
|
|
|
|
const decimals =
|
|
tokenDetails?.decimals && Number(tokenDetails.decimals?.toString(10));
|
|
|
|
if (tokenDetails?.standard === ERC20) {
|
|
tokenId = undefined;
|
|
}
|
|
|
|
// else if not a collectible already in state or standard === ERC20 return tokenDetails and tokenId
|
|
return {
|
|
tokenAmount,
|
|
toAddress,
|
|
decimals,
|
|
tokenId,
|
|
...tokenDetails,
|
|
};
|
|
}
|