129 lines
5.6 KiB
TypeScript
129 lines
5.6 KiB
TypeScript
import { Web3Wrapper } from '@0x/web3-wrapper';
|
|
import * as _ from 'lodash';
|
|
import { Dispatch } from 'redux';
|
|
|
|
import { BIG_NUMBER_ZERO, SUPPORTED_TOKEN_ASSET_DATA_WITH_BRIDGE_ORDERS } from '../constants';
|
|
import { AccountState, BaseCurrency, OrderProcessState, ProviderState, QuoteFetchOrigin } from '../types';
|
|
import { analytics } from '../util/analytics';
|
|
import { assetUtils } from '../util/asset';
|
|
import { coinbaseApi } from '../util/coinbase_api';
|
|
import { errorFlasher } from '../util/error_flasher';
|
|
import { errorReporter } from '../util/error_reporter';
|
|
import { swapQuoteUpdater } from '../util/swap_quote_updater';
|
|
|
|
import { actions } from './actions';
|
|
import { State } from './reducer';
|
|
|
|
export const asyncData = {
|
|
fetchEthPriceAndDispatchToStore: async (dispatch: Dispatch) => {
|
|
try {
|
|
const ethUsdPrice = await coinbaseApi.getEthUsdPrice();
|
|
dispatch(actions.updateEthUsdPrice(ethUsdPrice));
|
|
} catch (e) {
|
|
const errorMessage = 'Error fetching ETH/USD price';
|
|
errorFlasher.flashNewErrorMessage(dispatch, errorMessage);
|
|
dispatch(actions.updateEthUsdPrice(BIG_NUMBER_ZERO));
|
|
dispatch(actions.updateBaseCurrency(BaseCurrency.ETH));
|
|
errorReporter.report(e);
|
|
analytics.trackUsdPriceFailed();
|
|
}
|
|
},
|
|
fetchAvailableAssetDatasAndDispatchToStore: async (state: State, dispatch: Dispatch) => {
|
|
const { providerState, assetMetaDataMap, network } = state;
|
|
const swapQuoter = providerState.swapQuoter;
|
|
try {
|
|
const wethAssetData = await swapQuoter.getEtherTokenAssetDataOrThrowAsync();
|
|
const assetDatas = await swapQuoter.getAvailableMakerAssetDatasAsync(wethAssetData);
|
|
const deduplicatedAssetDatas = _.uniq(assetDatas);
|
|
const assetsWithNativeOrders = assetUtils.createAssetsFromAssetDatas(
|
|
deduplicatedAssetDatas,
|
|
assetMetaDataMap,
|
|
network,
|
|
);
|
|
const assetsWithBridgeOrders = assetUtils.createAssetsFromAssetDatas(
|
|
SUPPORTED_TOKEN_ASSET_DATA_WITH_BRIDGE_ORDERS,
|
|
assetMetaDataMap,
|
|
network,
|
|
);
|
|
const assets = _.uniqBy(_.concat(assetsWithNativeOrders, assetsWithBridgeOrders), asset => asset.assetData);
|
|
dispatch(actions.setAvailableAssets(assets));
|
|
} catch (e) {
|
|
const errorMessage = 'Could not find any assets';
|
|
errorFlasher.flashNewErrorMessage(dispatch, errorMessage);
|
|
// On error, just specify that none are available
|
|
dispatch(actions.setAvailableAssets([]));
|
|
errorReporter.report(e);
|
|
}
|
|
},
|
|
fetchAccountInfoAndDispatchToStore: async (
|
|
providerState: ProviderState,
|
|
dispatch: Dispatch,
|
|
shouldAttemptUnlock: boolean = false,
|
|
shouldSetToLoading: boolean = false,
|
|
) => {
|
|
const web3Wrapper = providerState.web3Wrapper;
|
|
const provider = providerState.provider;
|
|
if (shouldSetToLoading && providerState.account.state !== AccountState.Loading) {
|
|
dispatch(actions.setAccountStateLoading());
|
|
}
|
|
let availableAddresses: string[];
|
|
try {
|
|
// TODO(bmillman): Add support at the web3Wrapper level for calling `eth_requestAccounts` instead of calling enable here
|
|
const isPrivacyModeEnabled = (provider as any).enable !== undefined;
|
|
availableAddresses =
|
|
isPrivacyModeEnabled && shouldAttemptUnlock
|
|
? await (provider as any).enable()
|
|
: await web3Wrapper.getAvailableAddressesAsync();
|
|
} catch (e) {
|
|
analytics.trackAccountUnlockDenied();
|
|
dispatch(actions.setAccountStateLocked());
|
|
return;
|
|
}
|
|
if (!_.isEmpty(availableAddresses)) {
|
|
const activeAddress = availableAddresses[0];
|
|
dispatch(actions.setAccountStateReady(activeAddress));
|
|
// tslint:disable-next-line:no-floating-promises
|
|
asyncData.fetchAccountBalanceAndDispatchToStore(activeAddress, providerState.web3Wrapper, dispatch);
|
|
} else {
|
|
dispatch(actions.setAccountStateLocked());
|
|
}
|
|
},
|
|
fetchAccountBalanceAndDispatchToStore: async (address: string, web3Wrapper: Web3Wrapper, dispatch: Dispatch) => {
|
|
try {
|
|
const ethBalanceInWei = await web3Wrapper.getBalanceInWeiAsync(address);
|
|
dispatch(actions.updateAccountEthBalance({ address, ethBalanceInWei }));
|
|
} catch (e) {
|
|
errorReporter.report(e);
|
|
// leave balance as is
|
|
return;
|
|
}
|
|
},
|
|
fetchCurrentSwapQuoteAndDispatchToStore: async (
|
|
state: State,
|
|
dispatch: Dispatch,
|
|
fetchOrigin: QuoteFetchOrigin,
|
|
options: { updateSilently: boolean },
|
|
) => {
|
|
const { swapOrderState, providerState, selectedAsset, selectedAssetUnitAmount } = state;
|
|
const swapQuoter = providerState.swapQuoter;
|
|
if (
|
|
selectedAssetUnitAmount !== undefined &&
|
|
selectedAsset !== undefined &&
|
|
selectedAssetUnitAmount.isGreaterThan(BIG_NUMBER_ZERO) &&
|
|
swapOrderState.processState === OrderProcessState.None
|
|
) {
|
|
await swapQuoteUpdater.updateSwapQuoteAsync(
|
|
swapQuoter,
|
|
dispatch,
|
|
selectedAsset,
|
|
selectedAssetUnitAmount,
|
|
fetchOrigin,
|
|
{
|
|
setPending: !options.updateSilently,
|
|
dispatchErrors: !options.updateSilently,
|
|
},
|
|
);
|
|
}
|
|
},
|
|
};
|