MyCrypto/common/libs/nodes/tokenBalanceProxy.ts
HenryNguyen5 04eaa08d6c Shepherd MVP Integration (#1413)
* initial mvp

* First functioning pass

* Add token balance shim

* Add working web3 implementation

* Fix tests

* Fix tsc errors

* Implement token batch splitting

* Undo logger change

* Fix linting errors

* Revert makeconfig change

* Add typing to token proxy + use string interpolation

* Remove useless parameter

* Remove logging

* Use type coercion to fix proxied methods

* Update shepherd

* Update to typescript 2.8.1

* Fix merged typings

* Address PR comments

* replace myc-shepherd with mycrypto-shepherd
2018-04-06 15:52:48 -05:00

83 lines
2.6 KiB
TypeScript

import { Token } from 'shared/types/network';
import ERC20 from 'libs/erc20';
import { TokenValue } from 'libs/units';
import { IProvider } from 'mycrypto-shepherd/dist/lib/types';
export const tokenBalanceHandler: ProxyHandler<IProvider> = {
get(target, propKey) {
const tokenBalanceShim = (address: string, token: Token) => {
const sendCallRequest: (...rpcArgs: any[]) => Promise<string> = Reflect.get(
target,
'sendCallRequest'
);
return sendCallRequest({
to: token.address,
data: ERC20.balanceOf.encodeInput({ _owner: address })
})
.then(result => ({ balance: TokenValue(result), error: null }))
.catch(err => ({
balance: TokenValue('0'),
error: `Caught error: ${err}`
}));
};
const splitBatches = (address: string, tokens: Token[]) => {
const batchSize = 10;
type SplitBatch = { address: string; tokens: Token[] }[];
const splitBatch: SplitBatch = [];
for (let i = 0; i < tokens.length; i++) {
const arrIdx = Math.ceil((i + 1) / batchSize) - 1;
const token = tokens[i];
if (!splitBatch[arrIdx]) {
splitBatch.push({ address, tokens: [token] });
} else {
splitBatch[arrIdx] = { address, tokens: [...splitBatch[arrIdx].tokens, token] };
}
}
return splitBatch;
};
const tokenBalancesShim = (address: string, tokens: Token[]) => {
const sendCallRequests: (...rpcArgs: any[]) => Promise<string[]> = Reflect.get(
target,
'sendCallRequests'
);
return sendCallRequests(
tokens.map(t => ({
to: t.address,
data: ERC20.balanceOf.encodeInput({ _owner: address })
}))
).then(response =>
response.map(item => {
if (item) {
return {
balance: TokenValue(item),
error: null
};
} else {
return {
balance: TokenValue('0'),
error: 'Invalid object shape'
};
}
})
);
};
if (propKey.toString() === 'getTokenBalance') {
return (address: string, token: Token) => tokenBalanceShim(address, token);
} else if (propKey.toString() === 'getTokenBalances') {
return (address: string, tokens: Token[]) =>
Promise.all(
splitBatches(address, tokens).map(({ address: addr, tokens: tkns }) =>
tokenBalancesShim(addr, tkns)
)
).then(res => res.reduce((acc, curr) => [...acc, ...curr], []));
} else {
return Reflect.get(target, propKey);
}
}
};