mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-12-23 09:52:26 +01:00
3732c5f71e
ESLint rules have been added to enforce our JSDoc conventions. These rules were introduced by updating `@metamask/eslint-config` to v9. Some of the rules have been disabled because the effort to fix all lint errors was too high. It might be easiest to enable these rules one directory at a time, or one rule at a time. Most of the changes in this PR were a result of running `yarn lint:fix`. There were a handful of manual changes that seemed obvious and simple to make. Anything beyond that and the rule was left disabled.
229 lines
7.0 KiB
JavaScript
229 lines
7.0 KiB
JavaScript
import { cloneDeep } from 'lodash';
|
|
|
|
const version = 48;
|
|
|
|
/**
|
|
* 1. Delete NetworkController.settings
|
|
* 2a. Migrate NetworkController.provider to Rinkeby if set to type 'rpc' or
|
|
* 'localhost'.
|
|
* 2b. Re-key provider.rpcTarget to provider.rpcUrl
|
|
* 3. Add localhost network to frequentRpcListDetail.
|
|
* 4. Delete CachedBalancesController.cachedBalances
|
|
* 5. Convert transactions metamaskNetworkId to decimal if they are hex
|
|
* 6. Convert address book keys from decimal to hex
|
|
* 7. Delete localhost key in IncomingTransactionsController
|
|
* 8. Merge 'localhost' tokens into 'rpc' tokens
|
|
*/
|
|
export default {
|
|
version,
|
|
async migrate(originalVersionedData) {
|
|
const versionedData = cloneDeep(originalVersionedData);
|
|
versionedData.meta.version = version;
|
|
const state = versionedData.data;
|
|
versionedData.data = transformState(state);
|
|
return versionedData;
|
|
},
|
|
};
|
|
|
|
const hexRegEx = /^0x[0-9a-f]+$/iu;
|
|
const chainIdRegEx = /^0x[1-9a-f]+[0-9a-f]*$/iu;
|
|
|
|
function transformState(state = {}) {
|
|
// 1. Delete NetworkController.settings
|
|
delete state.NetworkController?.settings;
|
|
|
|
// 2. Migrate NetworkController.provider to Rinkeby or rename rpcTarget key
|
|
const provider = state.NetworkController?.provider || {};
|
|
const isCustomRpcWithInvalidChainId =
|
|
provider.type === 'rpc' &&
|
|
(typeof provider.chainId !== 'string' ||
|
|
!chainIdRegEx.test(provider.chainId));
|
|
if (isCustomRpcWithInvalidChainId || provider.type === 'localhost') {
|
|
state.NetworkController.provider = {
|
|
type: 'rinkeby',
|
|
rpcUrl: '',
|
|
chainId: '0x4',
|
|
nickname: '',
|
|
rpcPrefs: {},
|
|
ticker: 'ETH',
|
|
};
|
|
} else if (state.NetworkController?.provider) {
|
|
if ('rpcTarget' in state.NetworkController.provider) {
|
|
const rpcUrl = state.NetworkController.provider.rpcTarget;
|
|
state.NetworkController.provider.rpcUrl = rpcUrl;
|
|
}
|
|
delete state.NetworkController?.provider?.rpcTarget;
|
|
}
|
|
|
|
// 3. Add localhost network to frequentRpcListDetail.
|
|
if (!state.PreferencesController) {
|
|
state.PreferencesController = {};
|
|
}
|
|
if (!state.PreferencesController.frequentRpcListDetail) {
|
|
state.PreferencesController.frequentRpcListDetail = [];
|
|
}
|
|
state.PreferencesController.frequentRpcListDetail.unshift({
|
|
rpcUrl: 'http://localhost:8545',
|
|
chainId: '0x539',
|
|
ticker: 'ETH',
|
|
nickname: 'Localhost 8545',
|
|
rpcPrefs: {},
|
|
});
|
|
|
|
// 4. Delete CachedBalancesController.cachedBalances
|
|
delete state.CachedBalancesController?.cachedBalances;
|
|
|
|
// 5. Convert transactions metamaskNetworkId to decimal if they are hex
|
|
const transactions = state.TransactionController?.transactions;
|
|
if (Array.isArray(transactions)) {
|
|
transactions.forEach((transaction) => {
|
|
const metamaskNetworkId = transaction?.metamaskNetworkId;
|
|
if (
|
|
typeof metamaskNetworkId === 'string' &&
|
|
hexRegEx.test(metamaskNetworkId)
|
|
) {
|
|
transaction.metamaskNetworkId = parseInt(
|
|
metamaskNetworkId,
|
|
16,
|
|
).toString(10);
|
|
}
|
|
});
|
|
}
|
|
|
|
// 6. Convert address book keys from decimal to hex
|
|
const addressBook = state.AddressBookController?.addressBook || {};
|
|
Object.keys(addressBook).forEach((networkKey) => {
|
|
if (/^\d+$/iu.test(networkKey)) {
|
|
const chainId = `0x${parseInt(networkKey, 10).toString(16)}`;
|
|
updateChainIds(addressBook[networkKey], chainId);
|
|
|
|
if (addressBook[chainId]) {
|
|
mergeAddressBookKeys(addressBook, networkKey, chainId);
|
|
} else {
|
|
addressBook[chainId] = addressBook[networkKey];
|
|
}
|
|
delete addressBook[networkKey];
|
|
}
|
|
});
|
|
|
|
// 7. Delete localhost key in IncomingTransactionsController
|
|
delete state.IncomingTransactionsController
|
|
?.incomingTxLastFetchedBlocksByNetwork?.localhost;
|
|
|
|
// 8. Merge 'localhost' tokens into 'rpc' tokens
|
|
const accountTokens = state.PreferencesController?.accountTokens;
|
|
if (accountTokens) {
|
|
Object.keys(accountTokens).forEach((account) => {
|
|
const localhostTokens = accountTokens[account]?.localhost || [];
|
|
|
|
if (localhostTokens.length > 0) {
|
|
const rpcTokens = accountTokens[account].rpc || [];
|
|
|
|
if (rpcTokens.length > 0) {
|
|
accountTokens[account].rpc = mergeTokenArrays(
|
|
localhostTokens,
|
|
rpcTokens,
|
|
);
|
|
} else {
|
|
accountTokens[account].rpc = localhostTokens;
|
|
}
|
|
}
|
|
delete accountTokens[account]?.localhost;
|
|
});
|
|
}
|
|
|
|
return state;
|
|
}
|
|
|
|
/**
|
|
* Merges the two given keys for the given address book in place.
|
|
*
|
|
* @param addressBook
|
|
* @param networkKey
|
|
* @param chainIdKey
|
|
*/
|
|
function mergeAddressBookKeys(addressBook, networkKey, chainIdKey) {
|
|
const networkKeyEntries = addressBook[networkKey] || {};
|
|
// For the new entries, start by copying the existing entries for the chainId
|
|
const newEntries = { ...addressBook[chainIdKey] };
|
|
|
|
// For each address of the old/networkId key entries
|
|
Object.keys(networkKeyEntries).forEach((address) => {
|
|
if (newEntries[address] && typeof newEntries[address] === 'object') {
|
|
const mergedEntry = {};
|
|
|
|
// Collect all keys from both entries and merge the corresponding chainId
|
|
// entry with the networkId entry
|
|
new Set([
|
|
...Object.keys(newEntries[address]),
|
|
...Object.keys(networkKeyEntries[address] || {}),
|
|
]).forEach((key) => {
|
|
// Use non-empty value for the current key, if any
|
|
mergedEntry[key] =
|
|
newEntries[address][key] || networkKeyEntries[address]?.[key] || '';
|
|
});
|
|
|
|
newEntries[address] = mergedEntry;
|
|
} else if (
|
|
networkKeyEntries[address] &&
|
|
typeof networkKeyEntries[address] === 'object'
|
|
) {
|
|
// If there is no corresponding chainId entry, just use the networkId entry
|
|
// directly
|
|
newEntries[address] = networkKeyEntries[address];
|
|
}
|
|
});
|
|
|
|
addressBook[chainIdKey] = newEntries;
|
|
}
|
|
|
|
/**
|
|
* Updates the chainId key values to the given chainId in place for all values
|
|
* of the given networkEntries object.
|
|
*
|
|
* @param networkEntries
|
|
* @param chainId
|
|
*/
|
|
function updateChainIds(networkEntries, chainId) {
|
|
Object.values(networkEntries).forEach((entry) => {
|
|
if (entry && typeof entry === 'object') {
|
|
entry.chainId = chainId;
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Merges the two given, non-empty arrays of token objects and returns a new
|
|
* array.
|
|
*
|
|
* @param localhostTokens
|
|
* @param rpcTokens
|
|
* @returns {Array<Object>}
|
|
*/
|
|
function mergeTokenArrays(localhostTokens, rpcTokens) {
|
|
const localhostTokensMap = tokenArrayToMap(localhostTokens);
|
|
const rpcTokensMap = tokenArrayToMap(rpcTokens);
|
|
|
|
const mergedTokens = [];
|
|
new Set([
|
|
...Object.keys(localhostTokensMap),
|
|
...Object.keys(rpcTokensMap),
|
|
]).forEach((tokenAddress) => {
|
|
mergedTokens.push({
|
|
...localhostTokensMap[tokenAddress],
|
|
...rpcTokensMap[tokenAddress],
|
|
});
|
|
});
|
|
|
|
return mergedTokens;
|
|
|
|
function tokenArrayToMap(array) {
|
|
return array.reduce((map, token) => {
|
|
if (token?.address && typeof token?.address === 'string') {
|
|
map[token.address] = token;
|
|
}
|
|
return map;
|
|
}, {});
|
|
}
|
|
}
|