mirror of
https://github.com/status-im/MyCrypto.git
synced 2025-01-19 07:32:34 +00:00
5d4b36d453
* Refactor babel/types * Refactor entry point * Refactor actions * Refactor api * Full project refactor -- Broad type fixing sweep * - completely fix merge conflicts - handle various type errors * Add tslint to package.json * Dependency cleanup * Fix module resolution * Work on type definitions for untyped libs * progress commit * Add more definition typing * various type additions * Add unit types * Fix sagaiterator + unit types * various types added * additional type additions * Fix typing on Sagas * remove flowfixmes; swap translate for translateRaw * Get rid of contracts - awaiting Henry's contract PR * Remove contracts from routing * Fix most of actions/reducers * refactor actions directory structure * fix reducer action type imports * Fix most of type errors pre-actions refactor * fix action creator imports in containers * Refactor more * Refactor index of actions * fix action imports; use module level index export * package-lock.json updated * Use action types in props * Type up action creators * Fix most of connect errors * Typefixing progress * More types * Fix run-time errors * Caching improvements for webpack * Remove path resolve from webpack * Update non-breaking packages to latest version * Fix token typing * Remove unused color code * Fix wallet decrypt dispatch * Set redux-form related props/functions to ANY, since we're stripping it out later on * Revert BigNumber.js package changes * Extend window to custom object for Perf * Format Navigation * Typecase keystore errors as any (since we shouldnt touch this) * Push wallet context fix * - find/replace value->payload in swap - properly type swap state properties - extract inline reducer into reducer function * - type local storage retrieved items as generic * - bind all RPCClient methods with fat arrow * - reformat * Change to enums for constants * Change state into any * Fix swap errors * ensure that seconds are passed into state as integers * Fix rest of errors * use parseInt explicitly instead of type coercion * Fix derivation-checker, remove flow command, add tslint command, add tslint-react, tell travis to use tslint instead of flow. * Whoops, remove those tests. * Remove unsupported (yet) config option. * Fix precommit to target ts and tsx files. * Fix some errors, ignore some silly rules. * Revert jest to v19, use ts-jest and make all tests typescript. Fixes all but one. * Get rid of saga tests * Fix tslint errors
229 lines
6.6 KiB
TypeScript
229 lines
6.6 KiB
TypeScript
import Big, { BigNumber } from 'bignumber.js';
|
|
import { Token } from 'config/data';
|
|
import EthTx from 'ethereumjs-tx';
|
|
import { addHexPrefix, padToEven, toChecksumAddress } from 'ethereumjs-util';
|
|
import ERC20 from 'libs/erc20';
|
|
import { TransactionWithoutGas } from 'libs/messages';
|
|
import { RPCNode } from 'libs/nodes';
|
|
import { INode } from 'libs/nodes/INode';
|
|
import { Ether, toTokenUnit, UnitKey, Wei } from 'libs/units';
|
|
import { isValidETHAddress } from 'libs/validators';
|
|
import { stripHexPrefixAndLower, valueToHex } from 'libs/values';
|
|
import { IWallet } from 'libs/wallet';
|
|
import translate, { translateRaw } from 'translations';
|
|
|
|
export interface TransactionInput {
|
|
token?: Token | null;
|
|
unit: UnitKey;
|
|
value: string;
|
|
to: string;
|
|
data: string;
|
|
}
|
|
|
|
export interface BroadcastTransactionStatus {
|
|
isBroadcasting: boolean;
|
|
signedTx: string;
|
|
successfullyBroadcast: boolean;
|
|
}
|
|
|
|
export interface BaseTransaction {
|
|
to: string;
|
|
value: string;
|
|
data: string;
|
|
gasLimit: BigNumber | string;
|
|
gasPrice: Wei | string;
|
|
chainId: number;
|
|
}
|
|
|
|
export interface RawTransaction extends BaseTransaction {
|
|
nonce: string;
|
|
}
|
|
|
|
export interface ExtendedRawTransaction extends RawTransaction {
|
|
// non-standard, legacy
|
|
from: string;
|
|
}
|
|
|
|
export interface CompleteTransaction extends RawTransaction {
|
|
rawTx: string;
|
|
signedTx: string;
|
|
}
|
|
|
|
// Get useable fields from an EthTx object.
|
|
export function getTransactionFields(tx: EthTx) {
|
|
// For some crazy reason, toJSON spits out an array, not keyed values.
|
|
const [nonce, gasPrice, gasLimit, to, value, data, v, r, s] = tx.toJSON();
|
|
|
|
return {
|
|
// No value comes back as '0x', but most things expect '0x00'
|
|
value: value === '0x' ? '0x00' : value,
|
|
// If data is 0x, it might as well not be there
|
|
data: data === '0x' ? null : data,
|
|
// To address is unchecksummed, which could cause mismatches in comparisons
|
|
to: toChecksumAddress(to),
|
|
// Everything else is as-is
|
|
nonce,
|
|
gasPrice,
|
|
gasLimit,
|
|
v,
|
|
r,
|
|
s
|
|
};
|
|
}
|
|
|
|
export async function generateCompleteTransactionFromRawTransaction(
|
|
node: INode,
|
|
tx: ExtendedRawTransaction,
|
|
wallet: IWallet,
|
|
token: Token | null | undefined
|
|
): Promise<CompleteTransaction> {
|
|
const { to, data, gasLimit, gasPrice, from, chainId, nonce } = tx;
|
|
// Reject bad addresses
|
|
if (!isValidETHAddress(to)) {
|
|
throw new Error(translateRaw('ERROR_5'));
|
|
}
|
|
// Reject token transactions without data
|
|
if (token && !data) {
|
|
throw new Error('Tokens must be sent with data');
|
|
}
|
|
if (typeof gasLimit === 'string' || typeof gasPrice === 'string') {
|
|
throw Error('Gas Limit and Gas Price should be of type bignumber');
|
|
}
|
|
// Reject gas limit under 21000 (Minimum for transaction)
|
|
// Reject if limit over 5000000
|
|
// TODO: Make this dynamic, the limit shifts
|
|
if (gasLimit.lessThan(21000)) {
|
|
throw new Error('Gas limit must be at least 21000 for transactions');
|
|
}
|
|
// Reject gasLimit over 5000000gwei
|
|
if (gasLimit.greaterThan(5000000)) {
|
|
throw new Error(translateRaw('GETH_GasLimit'));
|
|
}
|
|
// Reject gasPrice over 1000gwei (1000000000000)
|
|
const gwei = new Big('1000000000000');
|
|
if (gasPrice.amount.greaterThan(gwei)) {
|
|
throw new Error(
|
|
'Gas price too high. Please contact support if this was not a mistake.'
|
|
);
|
|
}
|
|
// build gasCost by multiplying gasPrice * gasLimit
|
|
const gasCost: Wei = new Wei(gasPrice.amount.times(gasLimit));
|
|
// Ensure their balance exceeds the amount they're sending
|
|
let value;
|
|
let balance;
|
|
const ETHBalance: Wei = await node.getBalance(from);
|
|
if (token) {
|
|
value = new Big(ERC20.$transfer(tx.data).value);
|
|
balance = toTokenUnit(await node.getTokenBalance(tx.from, token), token);
|
|
} else {
|
|
value = new Big(tx.value);
|
|
balance = ETHBalance.amount;
|
|
}
|
|
if (value.gt(balance)) {
|
|
throw new Error(translateRaw('GETH_Balance'));
|
|
}
|
|
// ensure gas cost is not greaterThan current eth balance
|
|
// TODO check that eth balance is not lesser than txAmount + gasCost
|
|
if (gasCost.amount.gt(ETHBalance.amount)) {
|
|
throw new Error(
|
|
`gasCost: ${gasCost.amount} greaterThan ETHBalance: ${ETHBalance.amount}`
|
|
);
|
|
}
|
|
// Taken from v3's `sanitizeHex`, ensures that the value is a %2 === 0
|
|
// prefix'd hex value.
|
|
const cleanHex = hex => addHexPrefix(padToEven(stripHexPrefixAndLower(hex)));
|
|
const cleanedRawTx = {
|
|
nonce: cleanHex(nonce),
|
|
gasPrice: cleanHex(gasPrice.toString(16)),
|
|
gasLimit: cleanHex(gasLimit.toString(16)),
|
|
to: toChecksumAddress(cleanHex(to)),
|
|
value: token ? '0x00' : cleanHex(value.toString(16)),
|
|
data: data ? cleanHex(data) : '',
|
|
chainId: chainId || 1
|
|
};
|
|
// Sign the transaction
|
|
const rawTxJson = JSON.stringify(cleanedRawTx);
|
|
const signedTx = await wallet.signRawTransaction(cleanedRawTx);
|
|
// Repeat all of this shit for Flow typechecking. Sealed objects don't
|
|
// like spreads, so we have to be explicit.
|
|
return {
|
|
nonce: cleanedRawTx.nonce,
|
|
gasPrice: cleanedRawTx.gasPrice,
|
|
gasLimit: cleanedRawTx.gasLimit,
|
|
to: cleanedRawTx.to,
|
|
value: cleanedRawTx.value,
|
|
data: cleanedRawTx.data,
|
|
chainId: cleanedRawTx.chainId,
|
|
rawTx: rawTxJson,
|
|
signedTx
|
|
};
|
|
}
|
|
|
|
export async function formatTxInput(
|
|
wallet: IWallet,
|
|
{ token, unit, value, to, data }: TransactionInput
|
|
): Promise<TransactionWithoutGas> {
|
|
if (unit === 'ether') {
|
|
return {
|
|
to,
|
|
from: await wallet.getAddress(),
|
|
value: valueToHex(new Ether(value)),
|
|
data
|
|
};
|
|
} else {
|
|
if (!token) {
|
|
throw new Error('No matching token');
|
|
}
|
|
const bigAmount = new Big(value);
|
|
const ERC20Data = ERC20.transfer(to, bigAmount);
|
|
return {
|
|
to: token.address,
|
|
from: await wallet.getAddress(),
|
|
value: '0x0',
|
|
data: ERC20Data
|
|
};
|
|
}
|
|
}
|
|
|
|
export async function generateCompleteTransaction(
|
|
wallet: IWallet,
|
|
nodeLib: RPCNode,
|
|
gasPrice: Wei,
|
|
gasLimit: BigNumber,
|
|
chainId: number,
|
|
transactionInput: TransactionInput
|
|
): Promise<CompleteTransaction> {
|
|
const { token } = transactionInput;
|
|
const { from, to, value, data } = await formatTxInput(
|
|
wallet,
|
|
transactionInput
|
|
);
|
|
const transaction: ExtendedRawTransaction = {
|
|
nonce: await nodeLib.getTransactionCount(from),
|
|
from,
|
|
to,
|
|
gasLimit,
|
|
value,
|
|
data,
|
|
chainId,
|
|
gasPrice
|
|
};
|
|
return await generateCompleteTransactionFromRawTransaction(
|
|
nodeLib,
|
|
transaction,
|
|
wallet,
|
|
token
|
|
);
|
|
}
|
|
|
|
// TODO determine best place for helper function
|
|
export function getBalanceMinusGasCosts(
|
|
gasLimit: BigNumber,
|
|
gasPrice: Wei,
|
|
balance: Wei
|
|
): Ether {
|
|
const weiGasCosts = gasPrice.amount.times(gasLimit);
|
|
const weiBalanceMinusGasCosts = balance.amount.minus(weiGasCosts);
|
|
return new Ether(weiBalanceMinusGasCosts);
|
|
}
|