2017-06-30 03:03:11 +04:00
|
|
|
// @flow
|
2017-07-14 01:02:39 +04:00
|
|
|
import { takeEvery, call, apply, put, select, fork } from 'redux-saga/effects';
|
2017-07-04 03:59:27 +04:00
|
|
|
import type { Effect } from 'redux-saga/effects';
|
2017-07-14 01:02:39 +04:00
|
|
|
import { setWallet, setBalance, setTokenBalances } from 'actions/wallet';
|
2017-06-30 03:03:11 +04:00
|
|
|
import type { UnlockPrivateKeyAction } from 'actions/wallet';
|
2017-07-04 16:19:04 +04:00
|
|
|
import { showNotification } from 'actions/notifications';
|
|
|
|
import translate from 'translations';
|
2017-07-14 01:02:39 +04:00
|
|
|
import { PrivKeyWallet, BaseWallet } from 'libs/wallet';
|
|
|
|
import { BaseNode } from 'libs/nodes';
|
|
|
|
import { getNodeLib } from 'selectors/config';
|
|
|
|
import { getWalletInst, getTokens } from 'selectors/wallet';
|
|
|
|
import Big from 'big.js';
|
2017-06-30 03:03:11 +04:00
|
|
|
|
2017-07-14 01:02:39 +04:00
|
|
|
// FIXME MOVE ME
|
|
|
|
function padLeft(n: string, width: number, z: string = '0'): string {
|
|
|
|
return n.length >= width ? n : new Array(width - n.length + 1).join(z) + n;
|
|
|
|
}
|
|
|
|
|
|
|
|
function getEthCallData(to: string, method: string, args: string[]) {
|
|
|
|
return {
|
|
|
|
to,
|
|
|
|
data: method + args.map(a => padLeft(a, 64)).join()
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function* updateAccountBalance() {
|
|
|
|
const node: BaseNode = yield select(getNodeLib);
|
|
|
|
const wallet: ?BaseWallet = yield select(getWalletInst);
|
|
|
|
if (!wallet) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
let balance = yield apply(node, node.getBalance, [wallet.getAddress()]);
|
|
|
|
yield put(setBalance(balance));
|
|
|
|
}
|
|
|
|
|
|
|
|
function* updateTokenBalances() {
|
|
|
|
const node = yield select(getNodeLib);
|
|
|
|
const wallet: ?BaseWallet = yield select(getWalletInst);
|
|
|
|
const tokens = yield select(getTokens);
|
|
|
|
if (!wallet) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const requests = tokens.map(token =>
|
|
|
|
getEthCallData(token.address, '0x70a08231', [wallet.getNakedAddress()])
|
|
|
|
);
|
|
|
|
// FIXME handle errors
|
|
|
|
const tokenBalances = yield apply(node, node.ethCall, [requests]);
|
|
|
|
yield put(
|
|
|
|
setTokenBalances(
|
|
|
|
tokens.reduce((acc, t, i) => {
|
|
|
|
// FIXME
|
|
|
|
if (tokenBalances[i].error || tokenBalances[i].result === '0x') {
|
|
|
|
return acc;
|
|
|
|
}
|
2017-07-16 16:02:13 -05:00
|
|
|
let balance = Big(Number(tokenBalances[i].result)).div(
|
|
|
|
Big(10).pow(t.decimal)
|
|
|
|
); // definitely not safe
|
2017-07-14 01:02:39 +04:00
|
|
|
acc[t.symbol] = balance;
|
|
|
|
return acc;
|
|
|
|
}, {})
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function* updateBalances() {
|
|
|
|
yield fork(updateAccountBalance);
|
|
|
|
yield fork(updateTokenBalances);
|
2017-07-04 03:59:27 +04:00
|
|
|
}
|
|
|
|
|
2017-07-16 16:02:13 -05:00
|
|
|
export function* unlockPrivateKey(
|
|
|
|
action?: UnlockPrivateKeyAction
|
|
|
|
): Generator<Effect, void, any> {
|
2017-07-03 22:21:19 -05:00
|
|
|
if (!action) return;
|
2017-07-04 16:19:04 +04:00
|
|
|
let wallet = null;
|
2017-07-14 01:02:39 +04:00
|
|
|
|
2017-07-04 16:19:04 +04:00
|
|
|
try {
|
2017-07-16 16:02:13 -05:00
|
|
|
wallet = new PrivKeyWallet(Buffer.from(action.payload.key, 'hex'));
|
2017-07-04 16:19:04 +04:00
|
|
|
} catch (e) {
|
|
|
|
yield put(showNotification('danger', translate('INVALID_PKEY')));
|
|
|
|
return;
|
|
|
|
}
|
2017-07-14 01:02:39 +04:00
|
|
|
yield put(setWallet(wallet));
|
|
|
|
yield call(updateBalances);
|
2017-06-30 03:03:11 +04:00
|
|
|
}
|
|
|
|
|
2017-07-14 01:02:39 +04:00
|
|
|
export default function* walletSaga(): Generator<Effect | Effect[], void, any> {
|
|
|
|
// useful for development
|
|
|
|
yield call(updateBalances);
|
|
|
|
yield [
|
|
|
|
takeEvery('WALLET_UNLOCK_PRIVATE_KEY', unlockPrivateKey),
|
|
|
|
takeEvery('CUSTOM_TOKEN_ADD', updateTokenBalances)
|
|
|
|
];
|
2017-06-30 03:03:11 +04:00
|
|
|
}
|