1
0
mirror of https://github.com/kremalicious/metamask-extension.git synced 2024-11-27 04:46:10 +01:00
metamask-extension/app/scripts/controllers/transactions/lib/util.js

117 lines
3.7 KiB
JavaScript
Raw Normal View History

import { isValidAddress } from 'ethereumjs-util'
2021-01-21 07:06:41 +01:00
import { ethErrors } from 'eth-rpc-errors'
import { addHexPrefix } from '../../../lib/util'
import { TRANSACTION_STATUSES } from '../../../../../shared/constants/transaction'
2018-04-13 22:18:45 +02:00
const normalizers = {
from: (from) => addHexPrefix(from),
2020-11-03 00:41:28 +01:00
to: (to, lowerCase) =>
lowerCase ? addHexPrefix(to).toLowerCase() : addHexPrefix(to),
2020-02-15 21:34:12 +01:00
nonce: (nonce) => addHexPrefix(nonce),
value: (value) => addHexPrefix(value),
data: (data) => addHexPrefix(data),
gas: (gas) => addHexPrefix(gas),
gasPrice: (gasPrice) => addHexPrefix(gasPrice),
2018-04-13 22:18:45 +02:00
}
2018-04-19 20:29:26 +02:00
/**
* Normalizes the given txParams
* @param {Object} txParams - The transaction params
* @param {boolean} [lowerCase] - Whether to lowercase the 'to' address.
* Default: true
* @returns {Object} the normalized tx params
2018-04-13 22:18:45 +02:00
*/
2020-11-03 00:41:28 +01:00
export function normalizeTxParams(txParams, lowerCase = true) {
2018-04-13 22:18:45 +02:00
// apply only keys in the normalizers
const normalizedTxParams = {}
2018-04-19 20:29:26 +02:00
for (const key in normalizers) {
if (txParams[key]) {
normalizedTxParams[key] = normalizers[key](txParams[key], lowerCase)
}
2018-04-13 22:18:45 +02:00
}
return normalizedTxParams
}
/**
* Validates the given tx parameters
* @param {Object} txParams - the tx params
* @throws {Error} if the tx params contains invalid fields
2018-04-19 20:29:26 +02:00
*/
2020-11-03 00:41:28 +01:00
export function validateTxParams(txParams) {
if (!txParams || typeof txParams !== 'object' || Array.isArray(txParams)) {
throw ethErrors.rpc.invalidParams(
'Invalid transaction params: must be an object.',
)
}
if (!txParams.to && !txParams.data) {
throw ethErrors.rpc.invalidParams(
'Invalid transaction params: must specify "data" for contract deployments, or "to" (and optionally "data") for all other types of transactions.',
)
}
validateFrom(txParams)
validateRecipient(txParams)
if ('value' in txParams) {
const value = txParams.value.toString()
if (value.includes('-')) {
throw ethErrors.rpc.invalidParams(
`Invalid transaction value "${txParams.value}": not a positive number.`,
2020-11-03 00:41:28 +01:00
)
}
if (value.includes('.')) {
throw ethErrors.rpc.invalidParams(
`Invalid transaction value of "${txParams.value}": number must be in wei.`,
2020-11-03 00:41:28 +01:00
)
}
}
}
/**
* Validates the {@code from} field in the given tx params
* @param {Object} txParams
* @throws {Error} if the from address isn't valid
2018-04-19 20:29:26 +02:00
*/
2020-11-03 00:41:28 +01:00
export function validateFrom(txParams) {
if (!(typeof txParams.from === 'string')) {
throw ethErrors.rpc.invalidParams(
`Invalid "from" address "${txParams.from}": not a string.`,
)
}
if (!isValidAddress(txParams.from)) {
throw ethErrors.rpc.invalidParams('Invalid "from" address.')
}
}
/**
* Validates the {@code to} field in the given tx params
* @param {Object} txParams - the tx params
* @returns {Object} the tx params
* @throws {Error} if the recipient is invalid OR there isn't tx data
2018-04-19 20:29:26 +02:00
*/
2020-11-03 00:41:28 +01:00
export function validateRecipient(txParams) {
2018-04-10 23:53:40 +02:00
if (txParams.to === '0x' || txParams.to === null) {
if (txParams.data) {
delete txParams.to
} else {
throw ethErrors.rpc.invalidParams('Invalid "to" address.')
}
2018-04-10 23:53:40 +02:00
} else if (txParams.to !== undefined && !isValidAddress(txParams.to)) {
throw ethErrors.rpc.invalidParams('Invalid "to" address.')
}
return txParams
2018-04-10 23:53:40 +02:00
}
2018-04-19 20:29:26 +02:00
/**
* Returns a list of final states
* @returns {string[]} the states that can be considered final states
*/
2020-11-03 00:41:28 +01:00
export function getFinalStates() {
2018-04-19 20:29:26 +02:00
return [
TRANSACTION_STATUSES.REJECTED, // the user has responded no!
TRANSACTION_STATUSES.CONFIRMED, // the tx has been included in a block.
TRANSACTION_STATUSES.FAILED, // the tx failed for some reason, included on tx data.
TRANSACTION_STATUSES.DROPPED, // the tx nonce was already used
2018-04-19 20:29:26 +02:00
]
}