2018-02-16 16:57:23 +00:00
|
|
|
import { SagaIterator } from 'redux-saga';
|
2018-03-14 20:10:14 +00:00
|
|
|
import { put, select, apply, call, take, takeEvery } from 'redux-saga/effects';
|
|
|
|
import EthTx from 'ethereumjs-tx';
|
|
|
|
import { toChecksumAddress } from 'ethereumjs-util';
|
|
|
|
import {
|
|
|
|
setTransactionData,
|
|
|
|
FetchTransactionDataAction,
|
|
|
|
addRecentTransaction,
|
|
|
|
resetTransactionData,
|
|
|
|
TypeKeys
|
|
|
|
} from 'actions/transactions';
|
|
|
|
import {
|
|
|
|
TypeKeys as TxTypeKeys,
|
|
|
|
BroadcastTransactionQueuedAction,
|
|
|
|
BroadcastTransactionSucceededAction,
|
|
|
|
BroadcastTransactionFailedAction
|
|
|
|
} from 'actions/transaction';
|
|
|
|
import { getNodeLib, getNetworkConfig } from 'selectors/config';
|
|
|
|
import { getWalletInst } from 'selectors/wallet';
|
|
|
|
import { INode } from 'libs/nodes';
|
|
|
|
import { hexEncodeData } from 'libs/nodes/rpc/utils';
|
|
|
|
import { getTransactionFields } from 'libs/transaction';
|
|
|
|
import { TypeKeys as ConfigTypeKeys } from 'actions/config';
|
|
|
|
import { TransactionData, TransactionReceipt, SavedTransaction } from 'types/transactions';
|
|
|
|
import { NetworkConfig } from 'types/network';
|
|
|
|
import { AppState } from 'reducers';
|
2018-02-16 16:57:23 +00:00
|
|
|
|
|
|
|
export function* fetchTxData(action: FetchTransactionDataAction): SagaIterator {
|
|
|
|
const txhash = action.payload;
|
|
|
|
let data: TransactionData | null = null;
|
|
|
|
let receipt: TransactionReceipt | null = null;
|
|
|
|
let error: string | null = null;
|
|
|
|
|
|
|
|
const node: INode = yield select(getNodeLib);
|
|
|
|
|
|
|
|
// Fetch data and receipt separately, not in parallel. Receipt should only be
|
|
|
|
// fetched if the tx is mined, and throws if it's not, but that's not really
|
|
|
|
// an "error", since we'd still want to show the unmined tx data.
|
|
|
|
try {
|
|
|
|
data = yield apply(node, node.getTransactionByHash, [txhash]);
|
|
|
|
} catch (err) {
|
|
|
|
console.warn('Failed to fetch transaction data', err);
|
|
|
|
error = err.message;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data && data.blockHash) {
|
|
|
|
try {
|
|
|
|
receipt = yield apply(node, node.getTransactionReceipt, [txhash]);
|
|
|
|
} catch (err) {
|
|
|
|
console.warn('Failed to fetch transaction receipt', err);
|
|
|
|
receipt = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
yield put(setTransactionData({ txhash, data, receipt, error }));
|
|
|
|
}
|
|
|
|
|
2018-03-14 20:10:14 +00:00
|
|
|
export function* saveBroadcastedTx(action: BroadcastTransactionQueuedAction) {
|
|
|
|
const { serializedTransaction: txBuffer, indexingHash: txIdx } = action.payload;
|
|
|
|
|
|
|
|
const res: BroadcastTransactionSucceededAction | BroadcastTransactionFailedAction = yield take([
|
|
|
|
TxTypeKeys.BROADCAST_TRANSACTION_SUCCEEDED,
|
|
|
|
TxTypeKeys.BROADCAST_TRASACTION_FAILED
|
|
|
|
]);
|
|
|
|
|
|
|
|
// If our TX succeeded, save it and update the store.
|
|
|
|
if (
|
|
|
|
res.type === TxTypeKeys.BROADCAST_TRANSACTION_SUCCEEDED &&
|
|
|
|
res.payload.indexingHash === txIdx
|
|
|
|
) {
|
|
|
|
const tx = new EthTx(txBuffer);
|
|
|
|
const savableTx: SavedTransaction = yield call(
|
|
|
|
getSaveableTransaction,
|
|
|
|
tx,
|
|
|
|
res.payload.broadcastedHash
|
|
|
|
);
|
|
|
|
yield put(addRecentTransaction(savableTx));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Given a serialized transaction, return a transaction we could save in LS
|
|
|
|
export function* getSaveableTransaction(tx: EthTx, hash: string): SagaIterator {
|
|
|
|
const fields = getTransactionFields(tx);
|
|
|
|
let from: string = '';
|
|
|
|
let chainId: number = 0;
|
|
|
|
|
|
|
|
try {
|
|
|
|
// Signed transactions have these fields
|
|
|
|
from = hexEncodeData(tx.getSenderAddress());
|
|
|
|
chainId = fields.chainId;
|
|
|
|
} catch (err) {
|
|
|
|
// Unsigned transactions (e.g. web3) don't, so grab them from current state
|
|
|
|
const wallet: AppState['wallet']['inst'] = yield select(getWalletInst);
|
|
|
|
const network: NetworkConfig = yield select(getNetworkConfig);
|
|
|
|
|
|
|
|
chainId = network.chainId;
|
|
|
|
if (wallet) {
|
|
|
|
from = wallet.getAddressString();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const savableTx: SavedTransaction = {
|
|
|
|
hash,
|
|
|
|
from,
|
|
|
|
chainId,
|
|
|
|
to: toChecksumAddress(fields.to),
|
|
|
|
value: fields.value,
|
|
|
|
time: Date.now()
|
|
|
|
};
|
|
|
|
return savableTx;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function* resetTxData() {
|
|
|
|
yield put(resetTransactionData());
|
|
|
|
}
|
|
|
|
|
2018-02-16 16:57:23 +00:00
|
|
|
export default function* transactions(): SagaIterator {
|
|
|
|
yield takeEvery(TypeKeys.TRANSACTIONS_FETCH_TRANSACTION_DATA, fetchTxData);
|
2018-03-14 20:10:14 +00:00
|
|
|
yield takeEvery(TxTypeKeys.BROADCAST_TRANSACTION_QUEUED, saveBroadcastedTx);
|
2018-05-29 14:51:42 +00:00
|
|
|
yield takeEvery(ConfigTypeKeys.CONFIG_CHANGE_NODE_SUCCEEDED, resetTxData);
|
2018-02-16 16:57:23 +00:00
|
|
|
}
|