MyCrypto/common/features/handleMetaMaskPolling.ts
Connor Bryan c52a975aad Handle MetaMask network change (#1988)
* Poll for changes to MetaMask's network and reset the wallet if they don't match local

* Add tests for

* Add a test suite for handleMetaMaskPolling

* Add a userful comment and remove an unnecessary jest.mock

* Remove superfluous comment

* Change logout to a window reload

* Adjust verbiage of test case
2018-07-03 13:12:39 -05:00

53 lines
1.5 KiB
TypeScript

import { Store } from 'redux';
import { Web3Wallet } from 'libs/wallet';
import { AppState } from './reducers';
import * as configNetworksSelectors from './config/networks/selectors';
import { walletSelectors } from './wallet';
export const METAMASK_POLLING_INTERVAL: number = 1000;
export const getActualChainId = (): Promise<string> =>
new Promise((resolve, reject) => {
const { web3 } = window as any;
if (!web3) {
reject('Web3 not found.');
}
return web3.version.getNetwork(
(err: Error, network: string) => (err ? reject(err) : resolve(network))
);
});
/**
* @desc
* MetaMask no longer refreshes the page automatically on network change,
* so we must poll to ensure the network is the same as the locally stored version.
* @see https://medium.com/metamask/breaking-change-no-longer-reloading-pages-on-network-change-4a3e1fd2f5e7
*/
export default async function handleMetaMaskPolling(store: Store<AppState>): Promise<boolean> {
const state = store.getState();
try {
// Locally stored network.
const web3Wallet = walletSelectors.getWalletInst(state);
// MetaMask's actual network.
const actualChainId = await getActualChainId();
const actualNetwork = configNetworksSelectors.getNetworkByChainId(state, actualChainId);
if (web3Wallet && actualNetwork && (web3Wallet as Web3Wallet).network !== actualNetwork.id) {
window.location.reload();
return true;
}
} catch (error) {
window.location.reload();
return true;
}
return false;
}