const { addHexPrefix, isValidAddress, } = require('ethereumjs-util') /** @module */ module.exports = { normalizeTxParams, validateTxParams, validateFrom, validateRecipient, getFinalStates, } // functions that handle normalizing of that key in txParams const normalizers = { from: from => addHexPrefix(from).toLowerCase(), to: to => addHexPrefix(to).toLowerCase(), nonce: nonce => addHexPrefix(nonce), value: value => addHexPrefix(value), data: data => addHexPrefix(data), gas: gas => addHexPrefix(gas), gasPrice: gasPrice => addHexPrefix(gasPrice), } /** normalizes txParams @param txParams {object} @returns {object} normalized txParams */ function normalizeTxParams (txParams) { // apply only keys in the normalizers const normalizedTxParams = {} for (const key in normalizers) { if (txParams[key]) normalizedTxParams[key] = normalizers[key](txParams[key]) } return normalizedTxParams } /** validates txParams @param txParams {object} */ function validateTxParams (txParams) { validateFrom(txParams) validateRecipient(txParams) if ('value' in txParams) { const value = txParams.value.toString() if (value.includes('-')) { throw new Error(`Invalid transaction value of ${txParams.value} not a positive number.`) } if (value.includes('.')) { throw new Error(`Invalid transaction value of ${txParams.value} number must be in wei`) } } } /** validates the from field in txParams @param txParams {object} */ function validateFrom (txParams) { if (!(typeof txParams.from === 'string')) throw new Error(`Invalid from address ${txParams.from} not a string`) if (!isValidAddress(txParams.from)) throw new Error('Invalid from address') } /** validates the to field in txParams @param txParams {object} */ function validateRecipient (txParams) { if (txParams.to === '0x' || txParams.to === null) { if (txParams.data) { delete txParams.to } else { throw new Error('Invalid recipient address') } } else if (txParams.to !== undefined && !isValidAddress(txParams.to)) { throw new Error('Invalid recipient address') } return txParams } /** @returns an {array} of states that can be considered final */ function getFinalStates () { return [ 'rejected', // the user has responded no! 'confirmed', // the tx has been included in a block. 'failed', // the tx failed for some reason, included on tx data. 'dropped', // the tx nonce was already used ] }