From 57ca5d9a67dc6ffbc3cc0aedea48df553c82d9d6 Mon Sep 17 00:00:00 2001 From: David Walsh Date: Fri, 28 Jul 2023 11:25:48 -0500 Subject: [PATCH 01/25] Implement Network Menu Search (#19985) * WIP: Implement Network Menu Search * Maintain order, add tests * Remove unwanted locale * Fix duplicate import, better focus and item autofocus --- app/_locales/en/messages.json | 3 + .../network-list-item/network-list-item.js | 9 ++- .../network-list-menu/network-list-menu.js | 66 ++++++++++++++++++- .../network-list-menu.test.js | 14 +++- 4 files changed, 87 insertions(+), 5 deletions(-) diff --git a/app/_locales/en/messages.json b/app/_locales/en/messages.json index ea628a029d6b..fdd3a614d3e8 100644 --- a/app/_locales/en/messages.json +++ b/app/_locales/en/messages.json @@ -2538,6 +2538,9 @@ "noNFTs": { "message": "No NFTs yet" }, + "noNetworksFound": { + "message": "No networks found for the given search query" + }, "noSnaps": { "message": "You don't have any snaps installed." }, diff --git a/ui/components/multichain/network-list-item/network-list-item.js b/ui/components/multichain/network-list-item/network-list-item.js index 442568a691cc..2ff07e9b205f 100644 --- a/ui/components/multichain/network-list-item/network-list-item.js +++ b/ui/components/multichain/network-list-item/network-list-item.js @@ -47,6 +47,7 @@ export const NetworkListItem = ({ name, iconSrc, selected = false, + focus = true, onClick, onDeleteClick, }) => { @@ -54,10 +55,10 @@ export const NetworkListItem = ({ const networkRef = useRef(); useEffect(() => { - if (networkRef.current && selected) { + if (networkRef.current && focus) { networkRef.current.focus(); } - }, [networkRef, selected]); + }, [networkRef, focus]); return ( { const lineaMainnetReleased = useSelector(isLineaMainnetNetworkReleased); + const showSearch = nonTestNetworks.length > 3; + useEffect(() => { if (currentlyOnTestNetwork) { dispatch(setShowTestNetworks(currentlyOnTestNetwork)); } }, [dispatch, currentlyOnTestNetwork]); + const [searchQuery, setSearchQuery] = useState(''); + + let searchResults = [...nonTestNetworks]; + const isSearching = searchQuery !== ''; + + if (isSearching) { + const fuse = new Fuse(searchResults, { + threshold: 0.2, + location: 0, + distance: 100, + maxPatternLength: 32, + minMatchCharLength: 1, + shouldSort: true, + keys: ['nickname', 'chainId', 'ticker'], + }); + fuse.setCollection(searchResults); + const fuseResults = fuse.search(searchQuery); + // Ensure order integrity with original list + searchResults = searchResults.filter((network) => + fuseResults.includes(network), + ); + } const generateMenuItems = (desiredNetworks) => { return desiredNetworks.map((network, index) => { @@ -98,6 +127,7 @@ export const NetworkListMenu = ({ onClose }) => { iconSrc={network?.rpcPrefs?.imageUrl} key={`${network.id || network.chainId}-${index}`} selected={isCurrentNetwork} + focus={isCurrentNetwork && !showSearch} onClick={() => { dispatch(toggleNetworkMenu()); if (network.providerType) { @@ -162,8 +192,40 @@ export const NetworkListMenu = ({ onClose }) => { {t('networkMenuHeading')} <> + {showSearch ? ( + + setSearchQuery(e.target.value)} + clearButtonOnClick={() => setSearchQuery('')} + clearButtonProps={{ + size: Size.SM, + }} + inputProps={{ autoFocus: true }} + /> + + ) : null} - {generateMenuItems(nonTestNetworks)} + {searchResults.length === 0 && isSearching ? ( + + {t('noNetworksFound')} + + ) : ( + generateMenuItems(searchResults) + )} { it('displays important controls', () => { - const { getByText } = render(); + const { getByText, getByPlaceholderText } = render(); expect(getByText('Add network')).toBeInTheDocument(); expect(getByText('Show test networks')).toBeInTheDocument(); + expect(getByPlaceholderText('Search')).toBeInTheDocument(); }); it('renders mainnet item', () => { @@ -99,4 +100,15 @@ describe('NetworkListMenu', () => { ).textContent; expect(selectedNodeText).toStrictEqual('Custom Mainnet RPC'); }); + + it('narrows down search results', () => { + const { queryByText, getByPlaceholderText } = render(); + + expect(queryByText('Chain 5')).toBeInTheDocument(); + + const searchBox = getByPlaceholderText('Search'); + fireEvent.change(searchBox, { target: { value: 'Main' } }); + + expect(queryByText('Chain 5')).not.toBeInTheDocument(); + }); }); From 232f974b85ca815f9e22e073e345b1368335be29 Mon Sep 17 00:00:00 2001 From: Peter <53189696+PeterYinusa@users.noreply.github.com> Date: Fri, 28 Jul 2023 12:28:01 -0400 Subject: [PATCH 02/25] [skip e2e] document keyring flows (#20263) --- test/scenarios/2. keyring/reset a wallet.csv | 8 ++++++++ 1 file changed, 8 insertions(+) create mode 100644 test/scenarios/2. keyring/reset a wallet.csv diff --git a/test/scenarios/2. keyring/reset a wallet.csv b/test/scenarios/2. keyring/reset a wallet.csv new file mode 100644 index 000000000000..17cdf7fd4b84 --- /dev/null +++ b/test/scenarios/2. keyring/reset a wallet.csv @@ -0,0 +1,8 @@ +Step,Test Steps,Test Data,Expected Result,Notes +1,Open the extension.,,The Welcome Back screen is shown., +2,Proceed to the Forgot password flow.,,The extension opens in the expanded view., +3,Confirm your 12-word phrase.,12-word phrase.,, +4,Create a password for your MetaMask wallet.,password (8 characters min).,, +5,Proceed to Restore your wallet.,,"The Ether balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Ethereum Mainnet.", \ No newline at end of file From c741222f43ec25cd4c8ecad111eb9f1faf3b268e Mon Sep 17 00:00:00 2001 From: Peter <53189696+PeterYinusa@users.noreply.github.com> Date: Fri, 28 Jul 2023 12:28:10 -0400 Subject: [PATCH 03/25] [skip e2e] document onboarding flows (#20262) --- test/scenarios/1. onboarding/create a wallet.csv | 16 ++++++++++++++++ test/scenarios/1. onboarding/import a wallet.csv | 11 +++++++++++ 2 files changed, 27 insertions(+) create mode 100644 test/scenarios/1. onboarding/create a wallet.csv create mode 100644 test/scenarios/1. onboarding/import a wallet.csv diff --git a/test/scenarios/1. onboarding/create a wallet.csv b/test/scenarios/1. onboarding/create a wallet.csv new file mode 100644 index 000000000000..e26408fba78b --- /dev/null +++ b/test/scenarios/1. onboarding/create a wallet.csv @@ -0,0 +1,16 @@ +Step,Test Steps,Test Data,Expected Result,Notes +1,Open the extension.,,The Welcome screen is shown., +2,Agree to the Terms of use.,,, +3,Proceed to Create a new wallet.,,, +4,Allow MetaMask to gather usage data.,,, +5,Create a password for your MetaMask wallet.,password (8 characters min).,, +6,Watch the secure your wallet video.,,"The video plays. +The video is audible. +Subtitles are shown in the video.", +7,Proceed with the recommended action to secure you wallet.,,, +8,Reveal your 12-word phrase.,,,Take a note of the 12-word phrase. +9,Confirm your 12-word phrase.,12-word phrase.,, +10,Proceed through the onboarding completion screens.,,, +11,Dismiss the Whats new dialog.,,"The Ether balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Ethereum Mainnet.", \ No newline at end of file diff --git a/test/scenarios/1. onboarding/import a wallet.csv b/test/scenarios/1. onboarding/import a wallet.csv new file mode 100644 index 000000000000..5ca4f9ef6735 --- /dev/null +++ b/test/scenarios/1. onboarding/import a wallet.csv @@ -0,0 +1,11 @@ +Step,Test Steps,Test Data,Expected Result,Notes +1,Open the extension.,,The Welcome screen is shown., +2,Agree to the Terms of use.,,, +3,Proceed to Import an existing wallet.,,, +4,Allow MetaMask to gather usage data.,,, +5,Confirm your 12-word phrase.,12-word phrase.,, +6,Create a password for your MetaMask wallet.,password (8 characters min).,, +7,Proceed through the onboarding completion screens.,,, +8,Dismiss the Whats new dialog.,,"The Ether balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Ethereum Mainnet.", \ No newline at end of file From 0efc63784deef97499512689393cefac44f63ced Mon Sep 17 00:00:00 2001 From: David Walsh Date: Fri, 28 Jul 2023 12:29:39 -0500 Subject: [PATCH 04/25] UX: Autofocus the Address field in Import NFT modal (#20225) --- ui/components/multichain/import-nfts-modal/import-nfts-modal.js | 1 - 1 file changed, 1 deletion(-) diff --git a/ui/components/multichain/import-nfts-modal/import-nfts-modal.js b/ui/components/multichain/import-nfts-modal/import-nfts-modal.js index 20e14d83f0d5..b399c0fb9560 100644 --- a/ui/components/multichain/import-nfts-modal/import-nfts-modal.js +++ b/ui/components/multichain/import-nfts-modal/import-nfts-modal.js @@ -237,7 +237,6 @@ export const ImportNftsModal = ({ onClose }) => { Date: Fri, 28 Jul 2023 19:08:59 +0100 Subject: [PATCH 05/25] Update transaction status to not be considered anonymous (#20049) * update transaction status to not be considered anonymous * fix failing e2e * revert status changes in txn spec file * Fix e2e by adding status --------- Co-authored-by: Olusegun Akintayo --- app/scripts/controllers/transactions/index.js | 3 ++- .../controllers/transactions/index.test.js | 18 +++++++++--------- test/e2e/metrics/transaction-finalized.spec.js | 2 ++ 3 files changed, 13 insertions(+), 10 deletions(-) diff --git a/app/scripts/controllers/transactions/index.js b/app/scripts/controllers/transactions/index.js index 5301d1b1c964..dac1ed40fd79 100644 --- a/app/scripts/controllers/transactions/index.js +++ b/app/scripts/controllers/transactions/index.js @@ -2385,10 +2385,12 @@ export default class TransactionController extends EventEmitter { uiCustomizations = null; } + /** The transaction status property is not considered sensitive and is now included in the non-anonymous event */ let properties = { chain_id: chainId, referrer, source, + status, network, eip_1559_version: eip1559Version, gas_edit_type: 'none', @@ -2410,7 +2412,6 @@ export default class TransactionController extends EventEmitter { } let sensitiveProperties = { - status, transaction_envelope_type: isEIP1559Transaction(txMeta) ? TRANSACTION_ENVELOPE_TYPE_NAMES.FEE_MARKET : TRANSACTION_ENVELOPE_TYPE_NAMES.LEGACY, diff --git a/app/scripts/controllers/transactions/index.test.js b/app/scripts/controllers/transactions/index.test.js index 1237c9be333d..be236657811d 100644 --- a/app/scripts/controllers/transactions/index.test.js +++ b/app/scripts/controllers/transactions/index.test.js @@ -2163,6 +2163,7 @@ describe('Transaction Controller', function () { device_model: 'N/A', transaction_speed_up: false, ui_customizations: null, + status: 'unapproved', }, sensitiveProperties: { default_gas: '0.000031501', @@ -2173,7 +2174,6 @@ describe('Transaction Controller', function () { transaction_replaced: undefined, first_seen: 1624408066355, transaction_envelope_type: TRANSACTION_ENVELOPE_TYPE_NAMES.LEGACY, - status: 'unapproved', }, }; @@ -2250,6 +2250,7 @@ describe('Transaction Controller', function () { device_model: 'N/A', transaction_speed_up: false, ui_customizations: null, + status: 'unapproved', }, sensitiveProperties: { default_gas: '0.000031501', @@ -2260,7 +2261,6 @@ describe('Transaction Controller', function () { transaction_replaced: undefined, first_seen: 1624408066355, transaction_envelope_type: TRANSACTION_ENVELOPE_TYPE_NAMES.LEGACY, - status: 'unapproved', }, }; @@ -2349,6 +2349,7 @@ describe('Transaction Controller', function () { device_model: 'N/A', transaction_speed_up: false, ui_customizations: null, + status: 'unapproved', }, sensitiveProperties: { default_gas: '0.000031501', @@ -2359,7 +2360,6 @@ describe('Transaction Controller', function () { transaction_replaced: undefined, first_seen: 1624408066355, transaction_envelope_type: TRANSACTION_ENVELOPE_TYPE_NAMES.LEGACY, - status: 'unapproved', }, }; @@ -2438,6 +2438,7 @@ describe('Transaction Controller', function () { device_model: 'N/A', transaction_speed_up: false, ui_customizations: null, + status: 'unapproved', }, sensitiveProperties: { default_gas: '0.000031501', @@ -2448,7 +2449,6 @@ describe('Transaction Controller', function () { transaction_replaced: undefined, first_seen: 1624408066355, transaction_envelope_type: TRANSACTION_ENVELOPE_TYPE_NAMES.LEGACY, - status: 'unapproved', }, }; @@ -2529,6 +2529,7 @@ describe('Transaction Controller', function () { device_model: 'N/A', transaction_speed_up: false, ui_customizations: null, + status: 'unapproved', }, sensitiveProperties: { gas_price: '2', @@ -2537,7 +2538,6 @@ describe('Transaction Controller', function () { transaction_replaced: undefined, first_seen: 1624408066355, transaction_envelope_type: TRANSACTION_ENVELOPE_TYPE_NAMES.LEGACY, - status: 'unapproved', }, }; await txController._trackTransactionMetricsEvent( @@ -2601,6 +2601,7 @@ describe('Transaction Controller', function () { device_model: 'N/A', transaction_speed_up: false, ui_customizations: null, + status: 'unapproved', }, sensitiveProperties: { baz: 3.0, @@ -2611,7 +2612,6 @@ describe('Transaction Controller', function () { transaction_replaced: undefined, first_seen: 1624408066355, transaction_envelope_type: TRANSACTION_ENVELOPE_TYPE_NAMES.LEGACY, - status: 'unapproved', }, }; @@ -2675,6 +2675,7 @@ describe('Transaction Controller', function () { device_model: 'N/A', transaction_speed_up: false, ui_customizations: ['flagged_as_malicious'], + status: 'unapproved', }, sensitiveProperties: { baz: 3.0, @@ -2685,7 +2686,6 @@ describe('Transaction Controller', function () { transaction_replaced: undefined, first_seen: 1624408066355, transaction_envelope_type: TRANSACTION_ENVELOPE_TYPE_NAMES.LEGACY, - status: 'unapproved', }, }; @@ -2749,6 +2749,7 @@ describe('Transaction Controller', function () { device_model: 'N/A', transaction_speed_up: false, ui_customizations: ['flagged_as_safety_unknown'], + status: 'unapproved', }, sensitiveProperties: { baz: 3.0, @@ -2759,7 +2760,6 @@ describe('Transaction Controller', function () { transaction_replaced: undefined, first_seen: 1624408066355, transaction_envelope_type: TRANSACTION_ENVELOPE_TYPE_NAMES.LEGACY, - status: 'unapproved', }, }; @@ -2831,6 +2831,7 @@ describe('Transaction Controller', function () { device_model: 'N/A', transaction_speed_up: false, ui_customizations: null, + status: 'unapproved', }, sensitiveProperties: { baz: 3.0, @@ -2842,7 +2843,6 @@ describe('Transaction Controller', function () { transaction_replaced: undefined, first_seen: 1624408066355, transaction_envelope_type: TRANSACTION_ENVELOPE_TYPE_NAMES.FEE_MARKET, - status: 'unapproved', estimate_suggested: GasRecommendations.medium, estimate_used: GasRecommendations.high, default_estimate: 'medium', diff --git a/test/e2e/metrics/transaction-finalized.spec.js b/test/e2e/metrics/transaction-finalized.spec.js index 32c245255a40..df4cc21d0e3a 100644 --- a/test/e2e/metrics/transaction-finalized.spec.js +++ b/test/e2e/metrics/transaction-finalized.spec.js @@ -263,6 +263,7 @@ describe('Transaction Finalized Event', function () { category: 'Transactions', locale: 'en', environment_type: 'background', + status: 'submitted', }, 'Transaction Submitted event without sensitive properties does not match the expected payload', ); @@ -317,6 +318,7 @@ describe('Transaction Finalized Event', function () { category: 'Transactions', locale: 'en', environment_type: 'background', + status: 'confirmed', }, 'Transaction Finalized event without sensitive properties does not match the expected payload', ); From 0c2519397debe157698e1f50d5f0a8fab165c643 Mon Sep 17 00:00:00 2001 From: Gauthier Petetin Date: Fri, 28 Jul 2023 15:15:18 -0300 Subject: [PATCH 06/25] feat: github action to check if PR has requested labels before being merged (#19984) * feat(action): check if pr includes requested labels * fix(action): add missing QA label * fix(action): check if no label prevents merging * fix(action): remove QA label check * fix(action): add missing reopened condition * fix(action): increase list of labels which prevent merges --- .../scripts/check-pr-has-required-labels.ts | 97 +++++++++++++++++++ .github/workflows/check-pr-labels.yml | 38 ++++++++ .github/workflows/do-not-merge.yml | 17 ---- package.json | 3 +- 4 files changed, 137 insertions(+), 18 deletions(-) create mode 100644 .github/scripts/check-pr-has-required-labels.ts create mode 100644 .github/workflows/check-pr-labels.yml delete mode 100644 .github/workflows/do-not-merge.yml diff --git a/.github/scripts/check-pr-has-required-labels.ts b/.github/scripts/check-pr-has-required-labels.ts new file mode 100644 index 000000000000..7e9ec3573efa --- /dev/null +++ b/.github/scripts/check-pr-has-required-labels.ts @@ -0,0 +1,97 @@ +import * as core from '@actions/core'; +import { context, getOctokit } from '@actions/github'; +import { GitHub } from '@actions/github/lib/utils'; + +main().catch((error: Error): void => { + console.error(error); + process.exit(1); +}); + +async function main(): Promise { + // "GITHUB_TOKEN" is an automatically generated, repository-specific access token provided by GitHub Actions. + const githubToken = process.env.GITHUB_TOKEN; + if (!githubToken) { + core.setFailed('GITHUB_TOKEN not found'); + process.exit(1); + } + + // Initialise octokit, required to call Github GraphQL API + const octokit: InstanceType = getOctokit(githubToken); + + // Retrieve pull request info from context + const prRepoOwner = context.repo.owner; + const prRepoName = context.repo.repo; + const prNumber = context.payload.pull_request?.number; + if (!prNumber) { + core.setFailed('Pull request number not found'); + process.exit(1); + } + + // Retrieve pull request labels + const prLabels = await retrievePullRequestLabels(octokit, prRepoOwner, prRepoName, prNumber); + + const preventMergeLabels = ["needs-qa", "QA'd but questions", "issues-found", "need-ux-ds-review", "blocked", "stale", "DO-NOT-MERGE"]; + + let hasTeamLabel = false; + + // Check pull request has at least required QA label and team label + for (const label of prLabels) { + if (label.startsWith("team-") || label === "external-contributor") { + console.log(`PR contains a team label as expected: ${label}`); + hasTeamLabel = true; + } + if (preventMergeLabels.includes(label)) { + throw new Error(`PR cannot be merged because it still contains this label: ${label}`); + } + if (hasTeamLabel) { + return; + } + } + + // Otherwise, throw an arror to prevent from merging + let errorMessage = ''; + if (!hasTeamLabel) { + errorMessage += 'No team labels found on the PR. '; + } + errorMessage += 'Please add the required label(s) before merging the PR.'; + throw new Error(errorMessage); + +} + +// This function retrieves the pull request on a specific repo +async function retrievePullRequestLabels(octokit: InstanceType, repoOwner: string, repoName: string, prNumber: number): Promise { + + const retrievePullRequestLabelsQuery = ` + query RetrievePullRequestLabels($repoOwner: String!, $repoName: String!, $prNumber: Int!) { + repository(owner: $repoOwner, name: $repoName) { + pullRequest(number: $prNumber) { + labels(first: 100) { + nodes { + name + } + } + } + } + } + `; + + const retrievePullRequestLabelsResult: { + repository: { + pullRequest: { + labels: { + nodes: { + name: string; + }[]; + } + }; + }; + } = await octokit.graphql(retrievePullRequestLabelsQuery, { + repoOwner, + repoName, + prNumber, + }); + + const pullRequestLabels = retrievePullRequestLabelsResult?.repository?.pullRequest?.labels?.nodes?.map(labelObject => labelObject?.name); + + return pullRequestLabels || []; +} diff --git a/.github/workflows/check-pr-labels.yml b/.github/workflows/check-pr-labels.yml new file mode 100644 index 000000000000..47de292ca1b5 --- /dev/null +++ b/.github/workflows/check-pr-labels.yml @@ -0,0 +1,38 @@ +name: "Check PR has required labels" +on: + pull_request: + branches: + - main + types: + - opened + - reopened + - synchronize + - labeled + - unlabeled + +jobs: + check-pr-labels: + runs-on: ubuntu-latest + permissions: + pull-requests: read + + steps: + - name: Checkout repository + uses: actions/checkout@v3 + with: + fetch-depth: 0 # This is needed to checkout all branches + + - name: Set up Node.js + uses: actions/setup-node@v3 + with: + node-version-file: '.nvmrc' + cache: yarn + + - name: Install dependencies + run: yarn --immutable + + - name: Check PR has required labels + id: check-pr-has-required-labels + env: + GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} + run: npm run check-pr-has-required-labels diff --git a/.github/workflows/do-not-merge.yml b/.github/workflows/do-not-merge.yml deleted file mode 100644 index 1315a282b1a7..000000000000 --- a/.github/workflows/do-not-merge.yml +++ /dev/null @@ -1,17 +0,0 @@ -# Fails the pull request if it has the "DO-NOT-MERGE" label - -name: Check "DO-NOT-MERGE" label - -on: - pull_request: - types: [opened, reopened, labeled, unlabeled, synchronize] - -jobs: - do-not-merge: - runs-on: ubuntu-latest - if: ${{ contains(github.event.pull_request.labels.*.name, 'DO-NOT-MERGE') }} - steps: - - name: 'Check for label "DO-NOT-MERGE"' - run: | - echo 'This check fails PRs with the "DO-NOT-MERGE" label to block merging' - exit 1 diff --git a/package.json b/package.json index 3643055418cc..ca3fada5f362 100644 --- a/package.json +++ b/package.json @@ -96,7 +96,8 @@ "fitness-functions": "ts-node development/fitness-functions/index.ts", "generate-beta-commit": "node ./development/generate-beta-commit.js", "validate-branch-name": "validate-branch-name", - "add-release-label-to-pr-and-linked-issues": "ts-node ./.github/scripts/add-release-label-to-pr-and-linked-issues.ts" + "add-release-label-to-pr-and-linked-issues": "ts-node ./.github/scripts/add-release-label-to-pr-and-linked-issues.ts", + "check-pr-has-required-labels": "ts-node ./.github/scripts/check-pr-has-required-labels.ts" }, "resolutions": { "simple-update-notifier@^1.0.0": "^2.0.0", From 280fd5f7efab6d9489cc4152a117a6dbfbd672aa Mon Sep 17 00:00:00 2001 From: Ashis Kumar Pradhan <38760485+akp111@users.noreply.github.com> Date: Fri, 28 Jul 2023 23:56:16 +0530 Subject: [PATCH 07/25] added deprecation message and comment in ui/check-box (#20226) * added deprecation message and comment in ui/check-box * Small adjustments to component name and story docs --------- Co-authored-by: georgewrmarshall --- ui/components/ui/check-box/README.mdx | 4 ++-- ui/components/ui/check-box/check-box.component.js | 9 +++++++++ ui/components/ui/check-box/check-box.stories.js | 2 +- 3 files changed, 12 insertions(+), 3 deletions(-) diff --git a/ui/components/ui/check-box/README.mdx b/ui/components/ui/check-box/README.mdx index 0318006f5041..3a73e7cd5824 100644 --- a/ui/components/ui/check-box/README.mdx +++ b/ui/components/ui/check-box/README.mdx @@ -2,12 +2,12 @@ import { Story, Canvas, ArgsTable } from '@storybook/addon-docs'; import CheckBox from '.'; -# Checkbox +# Check Box(Deprecated) A small interactive box that can be toggled by the user to indicate an affirmative or negative choice. - + ## Props diff --git a/ui/components/ui/check-box/check-box.component.js b/ui/components/ui/check-box/check-box.component.js index 734c60238823..5c4b1b300d01 100644 --- a/ui/components/ui/check-box/check-box.component.js +++ b/ui/components/ui/check-box/check-box.component.js @@ -10,6 +10,15 @@ const CHECKBOX_STATE = { export const { CHECKED, INDETERMINATE, UNCHECKED } = CHECKBOX_STATE; +/** + * @deprecated The `` component has been deprecated in favor of the new `` component from the component-library. + * Please update your code to use the new `` component instead, which can be found at ui/components/component-library/checkbox/checkbox.tsx. + * You can find documentation for the new Checkbox component in the MetaMask Storybook: + * {@link https://metamask.github.io/metamask-storybook/?path=/docs/components-componentlibrary-checkbox--docs} + * If you would like to help with the replacement of the old CheckBox component, please submit a pull request against this GitHub issue: + * {@link /~https://github.com/MetaMask/metamask-extension/issues/20163} + */ + const CheckBox = ({ className, disabled, diff --git a/ui/components/ui/check-box/check-box.stories.js b/ui/components/ui/check-box/check-box.stories.js index 907d27de5704..2791db15720e 100644 --- a/ui/components/ui/check-box/check-box.stories.js +++ b/ui/components/ui/check-box/check-box.stories.js @@ -15,7 +15,7 @@ const checkboxOptions = { }; export default { - title: 'Components/UI/Check Box', + title: 'Components/UI/Check Box(Deprecated)', component: CheckBox, parameters: { From 537f1c7aee85b8e8e0769e6d0904cf6a97b28ba9 Mon Sep 17 00:00:00 2001 From: Pedro Figueiredo Date: Fri, 28 Jul 2023 19:57:06 +0100 Subject: [PATCH 08/25] feat: implement swap event metric e2e test (#20129) * implement swap event metric e2e test * fix lint error * clean up initial balance helpers * fix eslint warnings * Fix `token_to_amount` format to address firefox bug and refactor tests --- .eslintrc.js | 3 + .mocharc.js | 2 + app/scripts/controllers/transactions/index.js | 13 +- jest.config.js | 2 + shared/constants/metametrics.ts | 12 + test/e2e/helpers.js | 62 +- test/e2e/helpers.test.js | 53 ++ test/e2e/metrics/mock-data.js | 524 ++++++++++++++++ test/e2e/metrics/swaps.spec.js | 572 ++++++++++++++++++ test/e2e/mv3/multiple-restarts.spec.js | 11 +- ui/ducks/swaps/swaps.js | 24 +- 11 files changed, 1255 insertions(+), 23 deletions(-) create mode 100644 test/e2e/helpers.test.js create mode 100644 test/e2e/metrics/mock-data.js create mode 100644 test/e2e/metrics/swaps.spec.js diff --git a/.eslintrc.js b/.eslintrc.js index 4ef8de9bf01a..8244ca4abcb4 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -246,6 +246,7 @@ module.exports = { 'shared/**/*.test.js', 'ui/**/*.test.js', 'ui/__mocks__/*.js', + 'test/e2e/helpers.test.js', ], extends: ['@metamask/eslint-config-mocha'], rules: { @@ -271,10 +272,12 @@ module.exports = { 'app/scripts/migrations/*.test.js', 'app/scripts/platforms/*.test.js', 'development/**/*.test.js', + 'development/**/*.test.ts', 'shared/**/*.test.js', 'shared/**/*.test.ts', 'test/helpers/*.js', 'test/jest/*.js', + 'test/e2e/helpers.test.js', 'ui/**/*.test.js', 'ui/__mocks__/*.js', 'shared/lib/error-utils.test.js', diff --git a/.mocharc.js b/.mocharc.js index 2ed699e6b106..5728dbacabbd 100644 --- a/.mocharc.js +++ b/.mocharc.js @@ -9,6 +9,8 @@ module.exports = { './app/scripts/controllers/permissions/**/*.test.js', './app/scripts/controllers/mmi-controller.test.js', './app/scripts/constants/error-utils.test.js', + './development/fitness-functions/**/*.test.ts', + './test/e2e/helpers.test.js', ], recursive: true, require: ['test/env.js', 'test/setup.js'], diff --git a/app/scripts/controllers/transactions/index.js b/app/scripts/controllers/transactions/index.js index dac1ed40fd79..d030fd4dc72b 100644 --- a/app/scripts/controllers/transactions/index.js +++ b/app/scripts/controllers/transactions/index.js @@ -40,7 +40,10 @@ import { hexWEIToDecGWEI, } from '../../../../shared/modules/conversion.utils'; import { isSwapsDefaultTokenAddress } from '../../../../shared/modules/swaps.utils'; -import { MetaMetricsEventCategory } from '../../../../shared/constants/metametrics'; +import { + MetaMetricsEventCategory, + MetaMetricsEventName, +} from '../../../../shared/constants/metametrics'; import { CHAIN_ID_TO_GAS_LIMIT_BUFFER_MAP, NETWORK_TYPES, @@ -2128,7 +2131,7 @@ export default class TransactionController extends EventEmitter { ); this._trackMetaMetricsEvent({ - event: 'Swap Completed', + event: MetaMetricsEventName.SwapCompleted, category: MetaMetricsEventCategory.Swaps, sensitiveProperties: { ...txMeta.swapMetaData, @@ -2139,6 +2142,12 @@ export default class TransactionController extends EventEmitter { trade_gas_cost_in_eth: transactionsCost.tradeGasCostInEth, trade_and_approval_gas_cost_in_eth: transactionsCost.tradeAndApprovalGasCostInEth, + // Firefox and Chrome have different implementations of the APIs + // that we rely on for communication accross the app. On Chrome big + // numbers are converted into number strings, on firefox they remain + // Big Number objects. As such, we convert them here for both + // browsers. + token_to_amount: txMeta.swapMetaData.token_to_amount.toString(10), }, }); } diff --git a/jest.config.js b/jest.config.js index d4cc552664bd..9f4a2a858bc8 100644 --- a/jest.config.js +++ b/jest.config.js @@ -14,6 +14,7 @@ module.exports = { '/shared/**/*.(js|ts|tsx)', '/ui/**/*.(js|ts|tsx)', '/development/fitness-functions/**/*.test.(js|ts|tsx)', + '/test/e2e/helpers.test.js', ], coverageDirectory: './coverage', coveragePathIgnorePatterns: ['.stories.*', '.snap'], @@ -50,6 +51,7 @@ module.exports = { '/shared/**/*.test.(js|ts)', '/ui/**/*.test.(js|ts|tsx)', '/development/fitness-functions/**/*.test.(js|ts|tsx)', + '/test/e2e/helpers.test.js', ], testTimeout: 5500, // We have to specify the environment we are running in, which is jsdom. The diff --git a/shared/constants/metametrics.ts b/shared/constants/metametrics.ts index 32f8fc22d767..32d6ea0e680a 100644 --- a/shared/constants/metametrics.ts +++ b/shared/constants/metametrics.ts @@ -612,6 +612,18 @@ export enum MetaMetricsEventName { ActivityScreenOpened = 'Activity Screen Opened', WhatsNewViewed = `What's New Viewed`, WhatsNewClicked = `What's New Link Clicked`, + PrepareSwapPageLoaded = 'Prepare Swap Page Loaded', + QuotesRequested = 'Quotes Requested', + QuotesReceived = 'Quotes Received', + BestQuoteReviewed = 'Best Quote Reviewed', + AllAvailableQuotesOpened = 'All Available Quotes Opened', + SwapStarted = 'Swap Started', + TransactionAdded = 'Transaction Added', + TransactionSubmitted = 'Transaction Submitted', + TransactionApproved = 'Transaction Approved', + SwapCompleted = 'Swap Completed', + TransactionFinalized = 'Transaction Finalized', + ExitedSwaps = 'Exited Swaps', } export enum MetaMetricsEventAccountType { diff --git a/test/e2e/helpers.js b/test/e2e/helpers.js index 916074937142..536c13bbb420 100644 --- a/test/e2e/helpers.js +++ b/test/e2e/helpers.js @@ -589,9 +589,10 @@ function mockPhishingDetection(mockServer) { const PRIVATE_KEY = '0x7C9529A67102755B7E6102D6D950AC5D5863C98713805CEC576B945B15B71EAC'; -const generateETHBalance = (eth) => convertToHexValue(eth * 10 ** 18); +const convertETHToHexGwei = (eth) => convertToHexValue(eth * 10 ** 18); + const defaultGanacheOptions = { - accounts: [{ secretKey: PRIVATE_KEY, balance: generateETHBalance(25) }], + accounts: [{ secretKey: PRIVATE_KEY, balance: convertETHToHexGwei(25) }], }; const SERVICE_WORKER_URL = 'chrome://inspect/#service-workers'; @@ -654,7 +655,7 @@ const DEFAULT_GANACHE_OPTIONS = { accounts: [ { secretKey: DEFAULT_PRIVATE_KEY, - balance: generateETHBalance(25), + balance: convertETHToHexGwei(25), }, ], }; @@ -687,6 +688,10 @@ const logInWithBalanceValidation = async (driver, ganacheServer) => { await assertAccountBalanceForDOM(driver, ganacheServer); }; +async function sleepSeconds(sec) { + return new Promise((resolve) => setTimeout(resolve, sec * 1000)); +} + function roundToXDecimalPlaces(number, decimalPlaces) { return Math.round(number * 10 ** decimalPlaces) / 10 ** decimalPlaces; } @@ -699,8 +704,15 @@ function generateRandNumBetween(x, y) { return randomNumber; } -async function sleepSeconds(sec) { - return new Promise((resolve) => setTimeout(resolve, sec * 1000)); +function genRandInitBal(minETHBal = 10, maxETHBal = 100, decimalPlaces = 4) { + const initialBalance = roundToXDecimalPlaces( + generateRandNumBetween(minETHBal, maxETHBal), + decimalPlaces, + ); + + const initialBalanceInHex = convertETHToHexGwei(initialBalance); + + return { initialBalance, initialBalanceInHex }; } async function terminateServiceWorker(driver) { @@ -762,12 +774,44 @@ async function getEventPayloads(driver, mockedEndpoints, hasRequest = true) { } return isPending === !hasRequest; - }, 10000); + }, driver.timeout); const mockedRequests = []; for (const mockedEndpoint of mockedEndpoints) { mockedRequests.push(...(await mockedEndpoint.getSeenRequests())); } - return mockedRequests.map((req) => req.body.json.batch).flat(); + + return mockedRequests.map((req) => req.body.json?.batch).flat(); +} + +// Asserts that each request passes all assertions in one group of assertions, and the order does not matter. +function assertInAnyOrder(requests, assertions) { + // Clone the array to avoid mutating the original + const assertionsClone = [...assertions]; + + return ( + requests.every((request) => { + for (let a = 0; a < assertionsClone.length; a++) { + const assertionArray = assertionsClone[a]; + + const passed = assertionArray.reduce( + (acc, currAssertionFn) => currAssertionFn(request) && acc, + true, + ); + + if (passed) { + // Remove the used assertion array + assertionsClone.splice(a, 1); + // Exit the loop early since we found a matching assertion + return true; + } + } + + // No matching assertion found for this request + return false; + }) && + // Ensure all assertions were used + assertionsClone.length === 0 + ); } module.exports = { @@ -810,7 +854,7 @@ module.exports = { WALLET_PASSWORD, WINDOW_TITLES, DEFAULT_GANACHE_OPTIONS, - generateETHBalance, + convertETHToHexGwei, roundToXDecimalPlaces, generateRandNumBetween, sleepSeconds, @@ -823,4 +867,6 @@ module.exports = { onboardingRevealAndConfirmSRP, onboardingCompleteWalletCreation, onboardingPinExtension, + assertInAnyOrder, + genRandInitBal, }; diff --git a/test/e2e/helpers.test.js b/test/e2e/helpers.test.js new file mode 100644 index 000000000000..aec169a0c425 --- /dev/null +++ b/test/e2e/helpers.test.js @@ -0,0 +1,53 @@ +import { assertInAnyOrder } from './helpers'; + +describe('assertInAnyOrder()', () => { + it('returns true when all requests pass unique assertions', () => { + const requests = ['req1', 'req2', 'req3']; + const assertions = [ + [(req) => req === 'req1'], + [(req) => req === 'req2'], + [(req) => req === 'req3'], + ]; + expect(assertInAnyOrder(requests, assertions)).toBe(true); + }); + + it('returns true when all requests pass unique assertions independently of the order', () => { + const requests = ['req1', 'req2', 'req3']; + const assertions = [ + [(req) => req === 'req3'], + [(req) => req === 'req2'], + [(req) => req === 'req1'], + ]; + expect(assertInAnyOrder(requests, assertions)).toBe(true); + }); + + it('returns false when a request cannot pass any assertions', () => { + const requests = ['req1', 'req2', 'unknown']; + const assertions = [ + [(req) => req === 'req1'], + [(req) => req === 'req2'], + [(req) => req === 'req3'], + ]; + expect(assertInAnyOrder(requests, assertions)).toBe(false); + }); + + it('returns false when there are unused assertions', () => { + const requests = ['req1', 'req2']; + const assertions = [ + [(req) => req === 'req1'], + [(req) => req === 'req2'], + [(req) => req === 'req3'], + ]; + expect(assertInAnyOrder(requests, assertions)).toBe(false); + }); + + it('returns false when there are unused requests', () => { + const requests = ['req1', 'req2', 'req3', 'req4']; + const assertions = [ + [(req) => req === 'req1'], + [(req) => req === 'req2'], + [(req) => req === 'req3'], + ]; + expect(assertInAnyOrder(requests, assertions)).toBe(false); + }); +}); diff --git a/test/e2e/metrics/mock-data.js b/test/e2e/metrics/mock-data.js new file mode 100644 index 000000000000..cc776e09b6e3 --- /dev/null +++ b/test/e2e/metrics/mock-data.js @@ -0,0 +1,524 @@ +const TOKENS_API_MOCK_RESULT = [ + { + name: 'Ethereum', + symbol: 'ETH', + decimals: 18, + type: 'native', + iconUrl: + 'https://token.metaswap.codefi.network/assets/nativeCurrencyLogos/ethereum.svg', + coingeckoId: 'ethereum', + address: '0x0000000000000000000000000000000000000000', + occurrences: 100, + aggregators: [], + }, + { + address: '0x6b175474e89094c44da98b954eedeac495271d0f', + symbol: 'DAI', + decimals: 18, + name: 'Dai Stablecoin', + iconUrl: 'https://crypto.com/price/coin-data/icon/DAI/color_icon.png', + type: 'erc20', + aggregators: [ + 'aave', + 'bancor', + 'cmc', + 'cryptocom', + 'coinGecko', + 'oneInch', + 'pmm', + 'zerion', + 'lifi', + ], + occurrences: 9, + fees: { + '0xb0da5965d43369968574d399dbe6374683773a65': 0, + }, + storage: { + balance: 2, + }, + }, + { + address: '0xa0b86991c6218b36c1d19d4a2e9eb0ce3606eb48', + symbol: 'USDC', + decimals: 6, + name: 'USD Coin', + iconUrl: 'https://crypto.com/price/coin-data/icon/USDC/color_icon.png', + type: 'erc20', + aggregators: [ + 'aave', + 'bancor', + 'cryptocom', + 'coinGecko', + 'oneInch', + 'pmm', + 'zerion', + 'lifi', + ], + occurrences: 8, + fees: {}, + storage: { + balance: 9, + }, + }, + { + address: '0xc6bdb96e29c38dc43f014eed44de4106a6a8eb5f', + symbol: 'INUINU', + decimals: 18, + name: 'Inu Inu', + iconUrl: + 'https://assets.coingecko.com/coins/images/26391/thumb/logo_square_200.png?1657752596', + type: 'erc20', + aggregators: ['coinGecko'], + occurrences: 1, + }, +]; + +const TOP_ASSETS_API_MOCK_RESULT = [ + { + address: '0x0000000000000000000000000000000000000000', + symbol: 'ETH', + }, + { + address: '0x6b175474e89094c44da98b954eedeac495271d0f', + symbol: 'DAI', + }, + { + address: '0xa0b86991c6218b36c1d19d4a2e9eb0ce3606eb48', + symbol: 'USDC', + }, + { + address: '0xdac17f958d2ee523a2206206994597c13d831ec7', + symbol: 'USDT', + }, +]; + +const AGGREGATOR_METADATA_API_MOCK_RESULT = { + airswapLight: { + color: '#2B71FF', + title: 'AirSwap', + icon: "', + }, + bancor: { + color: '#c9c9c9', + title: 'Bancor', + icon: "', + }, + curve: { + color: '#24292E', + title: 'Curve', + icon: "', + }, + paraswap: { + color: '#0058D4', + title: 'Paraswap', + icon: "', + }, + zeroEx: { + color: '#000', + title: '0x API', + icon: "', + }, + oneInch: { + color: '#323232', + title: '1inch', + icon: "', + }, + uniswap: { + color: '#FFE9F4', + title: 'Uniswap', + icon: "', + }, +}; + +const GAS_PRICE_API_MOCK_RESULT = { + SafeGasPrice: '30', + ProposeGasPrice: '30', + FastGasPrice: '30', +}; + +const FEATURE_FLAGS_API_MOCK_RESULT = [ + { + ethereum: { + fallbackToV1: false, + mobileActive: true, + extensionActive: true, + }, + bsc: { + fallbackToV1: false, + mobileActive: true, + extensionActive: true, + }, + polygon: { + fallbackToV1: false, + mobileActive: true, + extensionActive: true, + }, + avalanche: { + fallbackToV1: false, + mobileActive: true, + extensionActive: true, + }, + smartTransactions: { + mobileActive: false, + extensionActive: false, + }, + updated_at: '2022-03-17T15:54:00.360Z', + }, +]; + +const NETWORKS_API_MOCK_RESULT = [ + { + active: true, + chainId: 1, + chainName: 'Ethereum Mainnet', + }, + { + active: true, + chainId: 3, + chainName: 'Ethereum Testnet Ropsten', + }, + { + active: true, + chainId: 4, + chainName: 'Ethereum Testnet Rinkeby', + }, + { + active: true, + chainId: 5, + chainName: 'Ethereum Testnet Görli', + }, + { + active: true, + chainId: 10, + chainName: 'Optimistic Ethereum', + }, + { + active: true, + chainId: 42, + chainName: 'Ethereum Testnet Kovan', + }, + { + active: true, + chainId: 69, + chainName: 'Optimistic Ethereum Testnet Kovan', + }, +]; + +const TRADES_API_MOCK_RESULT = [ + { + trade: { + data: '0x5f575529000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001bc16d674ec8000000000000000000000000000000000000000000000000000000000000000000c0000000000000000000000000000000000000000000000000000000000000001c616972737761704c696768743446656544796e616d696346697865640000000000000000000000000000000000000000000000000000000000000000000001a000000000000000000000000000000000000000000000000017738c5ea89d7ec50000000000000000000000000000000000000000000000000000000064b90bca000000000000000000000000bb289bc97591f70d8216462df40ed713011b968a0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f0000000000000000000000000000000000000000000000cef6ff7513f0d8000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001b8ca24ff16b4000000000000000000000000000000000000000000000000000000000000000001b2c84c103f4da9c8be2df3e3b5369e67c0c3f89be2d30edd6c3d77a585b006178435294c5a9a75b1d8f13cb971ef90ac4680d5b2dbcc80cec69c2b53a0a015ea90000000000000000000000000000000000000000000000000034cb175d5cc000000000000000000000000000f326e4de8f66a0bdc0970b79e0924e33c79f1915000000000000000000000000000000000000000000000000000000000000000000a3', + from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1', + value: '2000000000000000000', + to: '0x881D40237659C251811CEC9c364ef91dC08D300C', + }, + hasRoute: false, + sourceAmount: '2000000000000000000', + destinationAmount: '3817827352165064638464', + error: null, + sourceToken: '0x0000000000000000000000000000000000000000', + destinationToken: '0x6b175474e89094c44da98b954eedeac495271d0f', + maxGas: 300000, + averageGas: 150000, + estimatedRefund: 0, + approvalNeeded: null, + fetchTime: 2136, + aggregator: 'airswapV4', + aggType: 'RFQ', + fee: 0.743, + quoteRefreshSeconds: 30, + gasMultiplier: 1, + sourceTokenRate: 1, + destinationTokenRate: 0.00052031, + priceSlippage: { + ratio: 1.0067731251974976, + calculationError: '', + bucket: 'low', + sourceAmountInETH: 2, + destinationAmountInETH: 1.9864537496050048, + sourceAmountInNativeCurrency: 2, + destinationAmountInNativeCurrency: 1.9864537496050048, + }, + }, + { + trade: { + data: '0x5f575529000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001bc16d674ec8000000000000000000000000000000000000000000000000000000000000000000c000000000000000000000000000000000000000000000000000000000000000136f6e65496e6368563546656544796e616d69630000000000000000000000000000000000000000000000000000000000000000000000000000000000000001e000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006b175474e89094c44da98b954eedeac495271d0f0000000000000000000000000000000000000000000000001b83413f0b3640000000000000000000000000000000000000000000000000ca8b8dcb068e8a14ed0000000000000000000000000000000000000000000000000000000000000120000000000000000000000000000000000000000000000000003e2c284391c000000000000000000000000000f326e4de8f66a0bdc0970b79e0924e33c79f1915000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000a8e449022e0000000000000000000000000000000000000000000000001b83413f0b3640000000000000000000000000000000000000000000000000ca8b8dcb068e8a14ed00000000000000000000000000000000000000000000000000000000000000600000000000000000000000000000000000000000000000000000000000000001c0000000000000000000000060594a405d53811d3bc4766596efd80fd545a270ab4991fe00000000000000000000000000000000000000000000000000be', + from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1', + value: '2000000000000000000', + to: '0x881D40237659C251811CEC9c364ef91dC08D300C', + }, + hasRoute: false, + sourceAmount: '2000000000000000000', + destinationAmount: '3812549203736060477891', + error: null, + sourceToken: '0x0000000000000000000000000000000000000000', + destinationToken: '0x6b175474e89094c44da98b954eedeac495271d0f', + maxGas: 1000000, + averageGas: 560305, + estimatedRefund: 0, + approvalNeeded: null, + fetchTime: 496, + aggregator: 'oneInchV5', + aggType: 'AGG', + fee: 0.875, + quoteRefreshSeconds: 30, + gasMultiplier: 1.06, + sourceTokenRate: 1, + destinationTokenRate: 0.00052031, + priceSlippage: { + ratio: 1.0081462619020451, + calculationError: '', + bucket: 'low', + sourceAmountInETH: 2, + destinationAmountInETH: 1.9837074761959097, + sourceAmountInNativeCurrency: 2, + destinationAmountInNativeCurrency: 1.9837074761959097, + }, + }, + { + trade: { + data: '0x5f575529000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001bc16d674ec8000000000000000000000000000000000000000000000000000000000000000000c000000000000000000000000000000000000000000000000000000000000000147061726173776170563546656544796e616d696300000000000000000000000000000000000000000000000000000000000000000000000000000000000003a000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006b175474e89094c44da98b954eedeac495271d0f0000000000000000000000000000000000000000000000001b83413f0b3640000000000000000000000000000000000000000000000000ca8b8ea009e72325d00000000000000000000000000000000000000000000000000000000000000120000000000000000000000000000000000000000000000000003e2c284391c000000000000000000000000000dc838074d95c89a5c2cbf26984fedc9160b6162000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000264a6886da90000000000000000000000000000000000000000000000000000000000000020000000000000000000000000eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee0000000000000000000000006b175474e89094c44da98b954eedeac495271d0f000000000000000000000000e592427a0aece92de3edee1f18e0157c058615640000000000000000000000000000000000000000000000001b83413f0b3640000000000000000000000000000000000000000000000000ca8b8ea009e72325d00000000000000000000000000000000000000000000000ceadc08e682552e2aa01000000000000000000000000000000000000000000000000000000000040000000000000000000000000000000000000000000000000000000000064b95f9000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001c00000000000000000000000000000000000000000000000000000000000000220c28a7a0a5ca4419b9a63b7914d07da1f00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002bc02aaa39b223fe8d0a0e5c4f27ead9083c756cc20001f46b175474e89094c44da98b954eedeac495271d0f00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000e7', + from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1', + value: '2000000000000000000', + to: '0x881D40237659C251811CEC9c364ef91dC08D300C', + }, + hasRoute: false, + sourceAmount: '2000000000000000000', + destinationAmount: '3812549442726211543722', + error: null, + sourceToken: '0x0000000000000000000000000000000000000000', + destinationToken: '0x6b175474e89094c44da98b954eedeac495271d0f', + maxGas: 2750000, + averageGas: 637198, + estimatedRefund: 0, + approvalNeeded: null, + fetchTime: 603, + aggregator: 'paraswap', + aggType: 'AGG', + fee: 0.875, + quoteRefreshSeconds: 30, + gasMultiplier: 1.05, + sourceTokenRate: 1, + destinationTokenRate: 0.00052031, + priceSlippage: { + ratio: 1.0081461997275625, + calculationError: '', + bucket: 'low', + sourceAmountInETH: 2, + destinationAmountInETH: 1.983707600544875, + sourceAmountInNativeCurrency: 2, + destinationAmountInNativeCurrency: 1.983707600544875, + }, + }, + { + trade: { + data: '0x5f575529000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001bc16d674ec8000000000000000000000000000000000000000000000000000000000000000000c0000000000000000000000000000000000000000000000000000000000000000f706d6d46656544796e616d696376340000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000030000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006b175474e89094c44da98b954eedeac495271d0f0000000000000000000000000000000000000000000000001b87a9050abbc0000000000000000000000000000000000000000000000000cabad41cdf62203d9c00000000000000000000000000000000000000000000000000000000000001200000000000000000000000000000000000000000000000000039c462440c4000000000000000000000000000f326e4de8f66a0bdc0970b79e0924e33c79f1915000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001e00000000000000000000000006b175474e89094c44da98b954eedeac495271d0f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc20000000000000000000000000000000000000000000000ceddfd0356e18429f80000000000000000000000000000000000000000000000001b87a9050abbc000000000000000000000000000a69babef1ca67a37ffaf7a485dfff3382056e78c00000000000000000000000074de5d4fcbf63e00296fd95d33236b97940166310000000000000000000000005cfe73b6021e818b776b421b1c4db2474086a7e100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000064b90bc501ffffffffffffffffffffffffffffffffffffff371debb764b90b2f000000330000000000000000000000000000000000000000000000000000000000000003000000000000000000000000000000000000000000000000000000000000001b5f189918172e0d762be517bfc4fd5d699d89f41791499f277714e2626bf88bda40822e54a270b614588af2b7616e6f83eb07e9092b61637939e3e2c96e9cb3830000000000000000000000000000000000000000000000001b87a9050abbc0000083', + from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1', + value: '2000000000000000000', + to: '0x881D40237659C251811CEC9c364ef91dC08D300C', + }, + hasRoute: false, + sourceAmount: '2000000000000000000', + destinationAmount: '3816025224307343108600', + error: null, + sourceToken: '0x0000000000000000000000000000000000000000', + destinationToken: '0x6b175474e89094c44da98b954eedeac495271d0f', + maxGas: 405000, + averageGas: 209421, + estimatedRefund: 0, + approvalNeeded: null, + fetchTime: 315, + aggregator: 'pmm', + aggType: 'RFQ', + fee: 0.813, + quoteRefreshSeconds: 30, + gasMultiplier: 1, + sourceTokenRate: 1, + destinationTokenRate: 0.00052031, + priceSlippage: { + ratio: 1.007241957770323, + calculationError: '', + bucket: 'low', + sourceAmountInETH: 2, + destinationAmountInETH: 1.9855160844593538, + sourceAmountInNativeCurrency: 2, + destinationAmountInNativeCurrency: 1.9855160844593538, + }, + }, + { + trade: { + data: '0x5f575529000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001bc16d674ec8000000000000000000000000000000000000000000000000000000000000000000c0000000000000000000000000000000000000000000000000000000000000000c307846656544796e616d69630000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000030000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006b175474e89094c44da98b954eedeac495271d0f0000000000000000000000000000000000000000000000001b83413f0b3640000000000000000000000000000000000000000000000000ca893f423817bb851f0000000000000000000000000000000000000000000000000000000000000120000000000000000000000000000000000000000000000000003e2c284391c000000000000000000000000000f326e4de8f66a0bdc0970b79e0924e33c79f1915000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001c8706394d50000000000000000000000006b175474e89094c44da98b954eedeac495271d0f000000000000000000000000c02aaa39b223fe8d0a0e5c4f27ead9083c756cc20000000000000000000000000000000000000000000000ceab651effc4a000000000000000000000000000000000000000000000000000001b83413f0b364000000000000000000000000000bb289bc97591f70d8216462df40ed713011b968a00000000000000000000000000000000000000000000000000000000000000000000000000000000000000005cfe73b6021e818b776b421b1c4db2474086a7e10000000064b90b6e000000000000000000000000000000000000000064b90b300000000000000000000000000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000001b5e7ab859c3fcdc6a4f4dfca4d038eefd1393581898967ffd314609102e529bcf7d923520466015965be7496bc96f8b6382530216ff881c98373ccbac6e55472d869584cd00000000000000000000000011ededebf63bef0ea2d2d071bdf88f71543ec6fb000000000000000000000000000000000000000000000096a22737cc64b90b31000000000000000000000000000000000000000000000000000a', + from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1', + value: '2000000000000000000', + to: '0x881D40237659C251811CEC9c364ef91dC08D300C', + }, + hasRoute: false, + sourceAmount: '2000000000000000000', + destinationAmount: '3812379590821165400064', + error: null, + sourceToken: '0x0000000000000000000000000000000000000000', + destinationToken: '0x6b175474e89094c44da98b954eedeac495271d0f', + maxGas: 1650000, + averageGas: 411000, + estimatedRefund: 0, + approvalNeeded: null, + fetchTime: 1600, + aggregator: 'zeroEx', + aggType: 'AGG', + fee: 0.875, + quoteRefreshSeconds: 30, + gasMultiplier: 1.2, + sourceTokenRate: 1, + destinationTokenRate: 0.00052031, + priceSlippage: { + ratio: 1.0081903875499196, + calculationError: '', + bucket: 'low', + sourceAmountInETH: 2, + destinationAmountInETH: 1.9836192249001605, + sourceAmountInNativeCurrency: 2, + destinationAmountInNativeCurrency: 1.9836192249001605, + }, + }, + { + trade: { + data: '0x5f575529000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001bc16d674ec8000000000000000000000000000000000000000000000000000000000000000000c000000000000000000000000000000000000000000000000000000000000000136f70656e4f6365616e46656544796e616d696300000000000000000000000000000000000000000000000000000000000000000000000000000000000000020000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006b175474e89094c44da98b954eedeac495271d0f0000000000000000000000000000000000000000000000001b83413f0b3640000000000000000000000000000000000000000000000000ca99b075a3301b90920000000000000000000000000000000000000000000000000000000000000120000000000000000000000000000000000000000000000000003e2c284391c000000000000000000000000000f326e4de8f66a0bdc0970b79e0924e33c79f1915000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000c4bc80f1a800000000000000000000000074de5d4fcbf63e00296fd95d33236b97940166310000000000000000000000000000000000000000000000001b83413f0b3640000000000000000000000000000000000000000000000000ca99b075a3301b909200000000000000000000000000000000000000000000000000000000000000800000000000000000000000000000000000000000000000000000000000000001c000c800000000000000000060594a405d53811d3bc4766596efd80fd545a27000000000000000000000000000000000000000000000000000000000004d', + from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1', + value: '2000000000000000000', + to: '0x881D40237659C251811CEC9c364ef91dC08D300C', + }, + hasRoute: false, + sourceAmount: '2000000000000000000', + destinationAmount: '3813588554813041538761', + error: null, + sourceToken: '0x0000000000000000000000000000000000000000', + destinationToken: '0x6b175474e89094c44da98b954eedeac495271d0f', + maxGas: 1000000, + averageGas: 560305, + estimatedRefund: 0, + approvalNeeded: null, + fetchTime: 885, + aggregator: 'openOcean', + aggType: 'AGG', + fee: 0.875, + quoteRefreshSeconds: 30, + gasMultiplier: 1.1, + sourceTokenRate: 1, + destinationTokenRate: 0.00052031, + priceSlippage: { + ratio: 1.0078758695226133, + calculationError: '', + bucket: 'low', + sourceAmountInETH: 2, + destinationAmountInETH: 1.9842482609547736, + sourceAmountInNativeCurrency: 2, + destinationAmountInNativeCurrency: 1.9842482609547736, + }, + }, + { + trade: { + data: '0x5f575529000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001bc16d674ec8000000000000000000000000000000000000000000000000000000000000000000c0000000000000000000000000000000000000000000000000000000000000001268617368466c6f7746656544796e616d69630000000000000000000000000000000000000000000000000000000000000000000000000000000000000000038000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006b175474e89094c44da98b954eedeac495271d0f0000000000000000000000000000000000000000000000001b83413f0b3640000000000000000000000000000000000000000000000000cab6c6327a78735d300000000000000000000000000000000000000000000000000000000000000120000000000000000000000000000000000000000000000000003e2c284391c000000000000000000000000000f326e4de8f66a0bdc0970b79e0924e33c79f191500000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000244f02109290000000000000000000000000000000000000000000000000000000000000020000000000000000000000000b570895414d77b793b1989b3a0d7eb6b13716352000000000000000000000000111bb8c3542f2b92fb41b8d913c01d378843111100000000000000000000000074de5d4fcbf63e00296fd95d33236b97940166310000000000000000000000005cfe73b6021e818b776b421b1c4db2474086a7e100000000000000000000000000000000000000000000000000000000000000000000000000000000000000006b175474e89094c44da98b954eedeac495271d0f0000000000000000000000000000000000000000000000001b83413f0b3640000000000000000000000000000000000000000000000000001b83413f0b3640000000000000000000000000000000000000000000000000ced9d9ea5d9f7af1600000000000000000000000000000000000000000000000000000000064b90b610000000000000000000000000000000000000000000000000000018972d3b1762867616e64616c6674686562726f776e67786d786e69001a146cc976a68d000000000000000000000000000000000000000000000000000000000000000001a000000000000000000000000000000000000000000000000000000000000000415595107d161b075c0b19df2b42a9edf6aa20b767bbbd41d000543e13b831f462217694352c3edaf31191d6130af46d48003509f4f0eb1946adbfc8c3c00b551a1c00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000033', + from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1', + value: '2000000000000000000', + to: '0x881D40237659C251811CEC9c364ef91dC08D300C', + }, + hasRoute: false, + sourceAmount: '2000000000000000000', + destinationAmount: '3815727114848172700000', + error: null, + sourceToken: '0x0000000000000000000000000000000000000000', + destinationToken: '0x6b175474e89094c44da98b954eedeac495271d0f', + maxGas: 1000000, + averageGas: 560305, + estimatedRefund: 0, + approvalNeeded: null, + fetchTime: 601, + aggregator: 'hashFlow', + aggType: 'AGG', + fee: 0.875, + quoteRefreshSeconds: 30, + gasMultiplier: 1.1, + sourceTokenRate: 1, + destinationTokenRate: 0.00052031, + priceSlippage: { + ratio: 1.0073195124366736, + calculationError: '', + bucket: 'low', + sourceAmountInETH: 2, + destinationAmountInETH: 1.9853609751266528, + sourceAmountInNativeCurrency: 2, + destinationAmountInNativeCurrency: 1.9853609751266528, + }, + }, +]; + +const NETWORKS_2_API_MOCK_RESULT = { + active: true, + networkId: 1, + chainId: 1, + chainName: 'Ethereum Mainnet', + nativeCurrency: { + name: 'Ether', + symbol: 'ETH', + decimals: 18, + address: '0x0000000000000000000000000000000000000000', + }, + iconUrl: 'https://s3.amazonaws.com/airswap-token-images/ETH.png', + blockExplorerUrl: 'https://etherscan.io', + networkType: 'L1', + aggregators: [ + 'airswapV3', + 'airswapV4', + 'oneInchV4', + 'oneInchV5', + 'paraswap', + 'pmm', + 'zeroEx', + 'openOcean', + 'hashFlow', + 'wrappedNative', + ], + refreshRates: { + quotes: 30, + quotesPrefetching: 30, + stxGetTransactions: 10, + stxBatchStatus: 5, + stxStatusDeadline: 160, + stxMaxFeeMultiplier: 2, + }, + parameters: { + refreshRates: { + quotes: 30, + quotesPrefetching: 30, + stxGetTransactions: 10, + stxBatchStatus: 5, + }, + stxStatusDeadline: 160, + stxMaxFeeMultiplier: 2, + }, +}; + +module.exports = { + TOKENS_API_MOCK_RESULT, + TOP_ASSETS_API_MOCK_RESULT, + AGGREGATOR_METADATA_API_MOCK_RESULT, + GAS_PRICE_API_MOCK_RESULT, + FEATURE_FLAGS_API_MOCK_RESULT, + NETWORKS_API_MOCK_RESULT, + TRADES_API_MOCK_RESULT, + NETWORKS_2_API_MOCK_RESULT, +}; diff --git a/test/e2e/metrics/swaps.spec.js b/test/e2e/metrics/swaps.spec.js new file mode 100644 index 000000000000..ba7e7db5270c --- /dev/null +++ b/test/e2e/metrics/swaps.spec.js @@ -0,0 +1,572 @@ +const { strict: assert } = require('assert'); +const { toHex } = require('@metamask/controller-utils'); +const FixtureBuilder = require('../fixture-builder'); +const { + withFixtures, + generateGanacheOptions, + DEFAULT_GANACHE_OPTIONS, + unlockWallet, + getEventPayloads, + assertInAnyOrder, + genRandInitBal, +} = require('../helpers'); +const { + buildQuote, + reviewQuote, + waitForTransactionToComplete, + checkActivityTransaction, + changeExchangeRate, +} = require('../swaps/shared'); +const { + MetaMetricsEventCategory, + MetaMetricsEventName, +} = require('../../../shared/constants/metametrics'); +const { + TOKENS_API_MOCK_RESULT, + TOP_ASSETS_API_MOCK_RESULT, + AGGREGATOR_METADATA_API_MOCK_RESULT, + GAS_PRICE_API_MOCK_RESULT, + FEATURE_FLAGS_API_MOCK_RESULT, + NETWORKS_API_MOCK_RESULT, + TRADES_API_MOCK_RESULT, + NETWORKS_2_API_MOCK_RESULT, +} = require('./mock-data'); + +const numberOfSegmentRequests = 19; + +async function mockSegmentAndMetaswapRequests(mockServer) { + return [ + await mockServer + .forPost('https://api.segment.io/v1/batch') + .withJsonBodyIncluding({ + batch: [{ properties: { category: MetaMetricsEventCategory.Swaps } }], + }) + .times() + .thenCallback(() => ({ statusCode: 200 })), + await mockServer + .forGet('https://swap.metaswap.codefi.network/networks/1/tokens') + .thenCallback(() => ({ statusCode: 200, json: TOKENS_API_MOCK_RESULT })), + await mockServer + .forGet('https://swap.metaswap.codefi.network/networks/1/topAssets') + .thenCallback(() => ({ + statusCode: 200, + json: TOP_ASSETS_API_MOCK_RESULT, + })), + await mockServer + .forGet( + 'https://swap.metaswap.codefi.network/networks/1/aggregatorMetadata', + ) + .thenCallback(() => ({ + statusCode: 200, + json: AGGREGATOR_METADATA_API_MOCK_RESULT, + })), + await mockServer + .forGet('https://gas-api.metaswap.codefi.network/networks/1/gasPrices') + .thenCallback(() => ({ + statusCode: 200, + json: GAS_PRICE_API_MOCK_RESULT, + })), + await mockServer + .forGet('https://swap.metaswap.codefi.network/featureFlags') + .thenCallback(() => ({ + statusCode: 200, + json: FEATURE_FLAGS_API_MOCK_RESULT, + })), + await mockServer + .forGet('https://tx-insights.metaswap.codefi.network/networks') + .thenCallback(() => ({ + statusCode: 200, + json: NETWORKS_API_MOCK_RESULT, + })), + await mockServer + .forGet('https://swap.metaswap.codefi.network/networks/1/trades') + .thenCallback(() => ({ + statusCode: 200, + json: TRADES_API_MOCK_RESULT, + })), + await mockServer + .forGet('https://swap.metaswap.codefi.network/networks/1') + .thenCallback(() => ({ + statusCode: 200, + json: NETWORKS_2_API_MOCK_RESULT, + })), + await mockServer + .forGet('https://token-api.metaswap.codefi.network/token/1337') + .thenCallback(() => ({ + statusCode: 200, + json: {}, + })), + ]; +} + +describe('Swap Eth for another Token', function () { + it('Completes a Swap between ETH and DAI after changing initial rate', async function () { + const { initialBalanceInHex } = genRandInitBal(); + + await withFixtures( + { + fixtures: new FixtureBuilder() + .withMetaMetricsController({ + metaMetricsId: 'fake-metrics-id', + participateInMetaMetrics: true, + }) + .build(), + ganacheOptions: generateGanacheOptions({ + accounts: [ + { + secretKey: DEFAULT_GANACHE_OPTIONS.accounts[0].secretKey, + balance: initialBalanceInHex, + }, + ], + }), + title: this.test.title, + testSpecificMock: mockSegmentAndMetaswapRequests, + }, + async ({ driver, mockedEndpoint: mockedEndpoints }) => { + await driver.navigate(); + + await unlockWallet(driver); + + await getQuoteAndSwapTokens(driver); + + const metricsReqs = await assertReqsNumAndFilterMetrics( + driver, + mockedEndpoints, + ); + + await assertNavSwapButtonClickedEvent(metricsReqs); + + await assertPrepareSwapPageLoadedEvents(metricsReqs); + + await assertQuotesRequestedEvents(metricsReqs); + + await assertQuotesReceivedAndBestQuoteReviewedEvents(metricsReqs); + + await assertAllAvailableQuotesOpenedEvents(metricsReqs); + + await assertSwapStartedEvents(metricsReqs); + + await assertSwapCompletedEvents(metricsReqs); + + await assertExitedSwapsEvents(metricsReqs); + }, + ); + }); +}); + +async function getQuoteAndSwapTokens(driver) { + await buildQuote(driver, { + amount: 2, + swapTo: 'DAI', + }); + await reviewQuote(driver, { + amount: 2, + swapFrom: 'TESTETH', + swapTo: 'DAI', + }); + await changeExchangeRate(driver); + await reviewQuote(driver, { + amount: 2, + swapFrom: 'TESTETH', + swapTo: 'DAI', + skipCounter: true, + }); + await driver.clickElement({ text: 'Swap', tag: 'button' }); + await waitForTransactionToComplete(driver, { tokenName: 'DAI' }); + await checkActivityTransaction(driver, { + index: 0, + amount: '2', + swapFrom: 'TESTETH', + swapTo: 'DAI', + }); +} + +async function assertReqsNumAndFilterMetrics(driver, mockedEndpoints) { + const events = await getEventPayloads(driver, mockedEndpoints); + + const numberOfMetaswapRequests = 9; + assert.equal( + events.length, + numberOfSegmentRequests + numberOfMetaswapRequests, + ); + + const reqs = events.slice(0, numberOfSegmentRequests); + + return reqs; +} + +async function assertNavSwapButtonClickedEvent(reqs) { + assert.equal(reqs[0].event, MetaMetricsEventName.NavSwapButtonClicked); + assert.deepStrictEqual(reqs[0].properties, { + category: MetaMetricsEventCategory.Swaps, + chain_id: toHex(1337), + environment_type: 'fullscreen', + locale: 'en', + location: 'Main View', + text: 'Swap', + token_symbol: 'ETH', + }); +} + +async function assertPrepareSwapPageLoadedEvents(reqs) { + const assertionsReq1 = [ + (req) => req.event === MetaMetricsEventName.PrepareSwapPageLoaded, + (req) => Object.keys(req.properties).length === 7, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + + (req) => req.properties?.current_stx_enabled === false, + (req) => req.properties?.is_hardware_wallet === false, + (req) => req.properties?.stx_enabled === false, + ]; + + const assertionsReq2 = [ + (req) => req.event === MetaMetricsEventName.PrepareSwapPageLoaded, + (req) => Object.keys(req.properties).length === 4, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + ]; + + assert.ok( + assertInAnyOrder([reqs[1], reqs[2]], [assertionsReq1, assertionsReq2]), + 'assertPrepareSwapPageLoadedEvents(): reqs[1] and reqs[2] did not match what was expected', + ); +} + +async function assertQuotesRequestedEvents(reqs) { + const assertionsReq3 = [ + (req) => req.event === MetaMetricsEventName.QuotesRequested, + (req) => Object.keys(req.properties).length === 14, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + + (req) => req.properties?.anonymizedData === true, + (req) => req.properties?.current_stx_enabled === false, + (req) => req.properties?.custom_slippage === false, + (req) => req.properties?.is_hardware_wallet === false, + (req) => req.properties?.request_type === 'Order', + (req) => req.properties?.slippage === 2, + (req) => req.properties?.stx_enabled === false, + (req) => req.properties?.token_from === 'TESTETH', + (req) => req.properties?.token_from_amount === '2', + (req) => req.properties?.token_to === 'DAI', + ]; + + const assertionsReq4 = [ + (req) => req.event === MetaMetricsEventName.QuotesRequested, + (req) => Object.keys(req.properties).length === 4, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + ]; + + assert.ok( + assertInAnyOrder([reqs[3], reqs[4]], [assertionsReq3, assertionsReq4]), + 'assertQuotesRequestedEvents(): reqs[3] and reqs[4] did not match what was expected', + ); +} + +async function assertQuotesReceivedAndBestQuoteReviewedEvents(reqs) { + const assertionsReq5 = [ + (req) => req.event === MetaMetricsEventName.QuotesReceived, + (req) => Object.keys(req.properties).length === 18, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + + (req) => req.properties?.anonymizedData === true, + (req) => typeof req.properties?.available_quotes === 'number', + (req) => typeof req.properties?.best_quote_source === 'string', + (req) => req.properties?.current_stx_enabled === false, + (req) => req.properties?.custom_slippage === false, + (req) => req.properties?.is_hardware_wallet === false, + (req) => req.properties?.request_type === 'Order', + (req) => typeof req.properties?.response_time === 'number', + (req) => req.properties?.slippage === 2, + (req) => req.properties?.stx_enabled === false, + (req) => req.properties?.token_from === 'TESTETH', + (req) => req.properties?.token_from_amount === '2', + (req) => req.properties?.token_to === 'DAI', + (req) => typeof req.properties?.token_to_amount === 'string', + ]; + + const assertionsReq6 = [ + (req) => req.event === MetaMetricsEventName.QuotesReceived, + (req) => Object.keys(req.properties).length === 4, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + ]; + + const assertionsReq7 = [ + (req) => req.event === MetaMetricsEventName.BestQuoteReviewed, + (req) => Object.keys(req.properties).length === 17, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + + (req) => typeof req.properties?.available_quotes === 'number', + (req) => typeof req.properties?.best_quote_source === 'string', + (req) => req.properties?.current_stx_enabled === false, + (req) => req.properties?.custom_slippage === false, + (req) => req.properties?.is_hardware_wallet === false, + (req) => req.properties?.request_type === false, + (req) => req.properties?.slippage === 2, + (req) => req.properties?.stx_enabled === false, + (req) => req.properties?.token_from === 'TESTETH', + (req) => req.properties?.token_from_amount === '2', + (req) => req.properties?.token_to === 'DAI', + (req) => typeof req.properties?.token_to_amount === 'string', + ]; + + const assertionsReq8 = [ + (req) => req.event === MetaMetricsEventName.BestQuoteReviewed, + (req) => Object.keys(req.properties).length === 4, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + ]; + + // When running this test on Chrome in particular, reqs[5], reqs[6], reqs[7] + // and reqs[8] sometimes switch order so we bundled them together for the + // assertion + + assert.ok( + assertInAnyOrder( + [reqs[5], reqs[6], reqs[7], reqs[8]], + [assertionsReq5, assertionsReq6, assertionsReq7, assertionsReq8], + ), + 'assertQuotesReceivedAndBestQuoteReviewedEvents(): reqs[5], reqs[6], reqs[7] and reqs[8] did not match what was expected', + ); +} + +async function assertAllAvailableQuotesOpenedEvents(reqs) { + const assertionsReq9 = [ + (req) => req.event === MetaMetricsEventName.AllAvailableQuotesOpened, + (req) => Object.keys(req.properties).length === 18, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + + (req) => typeof req.properties?.available_quotes === 'number', + (req) => typeof req.properties?.best_quote_source === 'string', + (req) => req.properties?.current_stx_enabled === false, + (req) => req.properties?.custom_slippage === false, + (req) => req.properties?.is_hardware_wallet === false, + (req) => req.properties?.request_type === false, + (req) => req.properties?.slippage === 2, + (req) => req.properties?.stx_enabled === false, + (req) => req.properties?.token_from === 'TESTETH', + (req) => req.properties?.token_from_amount === '2', + (req) => req.properties?.token_to === 'DAI', + (req) => req.properties?.token_to === 'DAI', + (req) => req.properties?.other_quote_selected === false, + (req) => req.properties?.other_quote_selected_source === null, + (req) => typeof req.properties?.token_to_amount === 'string', + ]; + + const assertionsReq10 = [ + (req) => req.event === MetaMetricsEventName.AllAvailableQuotesOpened, + (req) => Object.keys(req.properties).length === 4, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + ]; + + assert.ok( + assertInAnyOrder([reqs[9], reqs[10]], [assertionsReq9, assertionsReq10]), + 'assertAllAvailableQuotesOpenedEvents(): reqs[9] and reqs[10] did not match what was expected', + ); +} + +async function assertSwapStartedEvents(reqs) { + const assertionsReq11 = [ + (req) => req.event === MetaMetricsEventName.SwapStarted, + (req) => Object.keys(req.properties).length === 24, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + + (req) => req.properties?.token_from === 'TESTETH', + (req) => req.properties?.token_from_amount === '2', + (req) => req.properties?.token_to === 'DAI', + (req) => req.properties?.slippage === 2, + (req) => req.properties?.custom_slippage === false, + (req) => req.properties?.is_hardware_wallet === false, + (req) => req.properties?.stx_enabled === false, + (req) => req.properties?.current_stx_enabled === false, + (req) => typeof req.properties?.token_to_amount === 'string', + (req) => typeof req.properties?.best_quote_source === 'string', + (req) => typeof req.properties?.other_quote_selected === 'boolean', + (req) => typeof req.properties?.gas_fees === 'string', + (req) => typeof req.properties?.estimated_gas === 'string', + (req) => typeof req.properties?.suggested_gas_price === 'string', + (req) => typeof req.properties?.reg_tx_fee_in_usd === 'number', + (req) => typeof req.properties?.reg_tx_fee_in_eth === 'number', + (req) => typeof req.properties?.reg_tx_max_fee_in_usd === 'number', + (req) => typeof req.properties?.reg_tx_max_fee_in_eth === 'number', + (req) => typeof req.properties?.other_quote_selected_source === 'string', + ]; + + const assertionsReq12 = [ + (req) => req.event === MetaMetricsEventName.SwapStarted, + (req) => Object.keys(req.properties).length === 4, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + ]; + + assert.ok( + assertInAnyOrder([reqs[11], reqs[12]], [assertionsReq11, assertionsReq12]), + 'assertSwapStartedEvents(): reqs[11] and reqs[12] did not match what was expected', + ); +} + +async function assertSwapCompletedEvents(reqs) { + const assertionsReq13 = [ + (req) => req.event === MetaMetricsEventName.SwapCompleted, + (req) => Object.keys(req.properties).length === 30, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'background', + (req) => req.properties?.locale === 'en', + + (req) => req.properties?.token_from === 'TESTETH', + (req) => req.properties?.token_from_amount === '2', + (req) => req.properties?.token_to === 'DAI', + (req) => typeof req.properties?.token_to_amount === 'string', + (req) => req.properties?.slippage === 2, + (req) => req.properties?.custom_slippage === false, + (req) => req.properties?.best_quote_source === 'airswapV4', + (req) => typeof req.properties?.other_quote_selected === 'boolean', + (req) => typeof req.properties?.other_quote_selected_source === 'string', + (req) => typeof req.properties?.gas_fees === 'string', + (req) => typeof req.properties?.estimated_gas === 'string', + (req) => req.properties?.suggested_gas_price === '30', + (req) => req.properties?.used_gas_price === '30', + (req) => req.properties?.is_hardware_wallet === false, + (req) => req.properties?.stx_enabled === false, + (req) => req.properties?.current_stx_enabled === false, + (req) => typeof req.properties?.reg_tx_fee_in_usd === 'number', + (req) => typeof req.properties?.reg_tx_fee_in_eth === 'number', + (req) => typeof req.properties?.reg_tx_max_fee_in_usd === 'number', + (req) => typeof req.properties?.reg_tx_max_fee_in_eth === 'number', + (req) => req.properties?.token_to_amount_received === '', + (req) => req.properties?.quote_vs_executionRatio === null, + (req) => req.properties?.estimated_vs_used_gasRatio === '100%', + (req) => req.properties?.approval_gas_cost_in_eth === 0, + (req) => typeof req.properties?.trade_gas_cost_in_eth === 'number', + (req) => + typeof req.properties?.trade_and_approval_gas_cost_in_eth === 'number', + ]; + + const assertionsReq14 = [ + (req) => req.event === MetaMetricsEventName.SwapCompleted, + (req) => Object.keys(req.properties).length === 4, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'background', + (req) => req.properties?.locale === 'en', + ]; + + assert.ok( + assertInAnyOrder([reqs[13], reqs[14]], [assertionsReq13, assertionsReq14]), + 'assertSwapCompletedEvents(): reqs[13] and reqs[14] did not match what was expected', + ); +} + +async function assertExitedSwapsEvents(reqs) { + const assertionsReq15 = [ + (req) => req.event === MetaMetricsEventName.ExitedSwaps, + (req) => Object.keys(req.properties).length === 12, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + + (req) => req.properties?.token_from_amount === '2', + (req) => req.properties?.request_type === false, + (req) => req.properties?.slippage === 2, + (req) => req.properties?.custom_slippage === false, + (req) => req.properties?.current_screen === 'awaiting-swap', + (req) => req.properties?.is_hardware_wallet === false, + (req) => req.properties?.stx_enabled === false, + (req) => req.properties?.current_stx_enabled === false, + ]; + + const assertionsReq16 = [ + (req) => req.event === MetaMetricsEventName.ExitedSwaps, + (req) => Object.keys(req.properties).length === 4, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + ]; + + assert.ok( + assertInAnyOrder([reqs[15], reqs[16]], [assertionsReq15, assertionsReq16]), + 'assertExitedSwapsEvents(): reqs[15] and reqs[16] did not match what was expected', + ); + + const assertionsReq17 = [ + (req) => req.event === MetaMetricsEventName.ExitedSwaps, + (req) => Object.keys(req.properties).length === 9, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + + (req) => req.properties?.custom_slippage === true, + (req) => req.properties?.current_screen === 'awaiting-swap', + (req) => req.properties?.is_hardware_wallet === false, + (req) => req.properties?.stx_enabled === false, + (req) => req.properties?.current_stx_enabled === false, + ]; + + const assertionsReq18 = [ + (req) => req.event === MetaMetricsEventName.ExitedSwaps, + (req) => Object.keys(req.properties).length === 4, + + (req) => req.properties?.category === MetaMetricsEventCategory.Swaps, + (req) => req.properties?.chain_id === toHex(1337), + (req) => req.properties?.environment_type === 'fullscreen', + (req) => req.properties?.locale === 'en', + ]; + + assert.ok( + assertInAnyOrder([reqs[17], reqs[18]], [assertionsReq17, assertionsReq18]), + 'assertExitedSwapsEvents(): reqs[17] and reqs[18] did not match what was expected', + ); +} diff --git a/test/e2e/mv3/multiple-restarts.spec.js b/test/e2e/mv3/multiple-restarts.spec.js index a90111d2ac77..9dad3a34ab48 100644 --- a/test/e2e/mv3/multiple-restarts.spec.js +++ b/test/e2e/mv3/multiple-restarts.spec.js @@ -6,22 +6,19 @@ const { WALLET_PASSWORD, WINDOW_TITLES, DEFAULT_GANACHE_OPTIONS, - generateETHBalance, - roundToXDecimalPlaces, generateRandNumBetween, sleepSeconds, terminateServiceWorker, unlockWallet, largeDelayMs, + genRandInitBal, + roundToXDecimalPlaces, } = require('../helpers'); const FixtureBuilder = require('../fixture-builder'); describe('MV3 - Restart service worker multiple times', function () { it('Simple simple send flow within full screen view should still be usable', async function () { - const initialBalance = roundToXDecimalPlaces( - generateRandNumBetween(10, 100), - 4, - ); + const { initialBalance, initialBalanceInHex } = genRandInitBal(); await withFixtures( { @@ -30,7 +27,7 @@ describe('MV3 - Restart service worker multiple times', function () { accounts: [ { secretKey: DEFAULT_GANACHE_OPTIONS.accounts[0].secretKey, - balance: generateETHBalance(initialBalance), + balance: initialBalanceInHex, }, ], }), diff --git a/ui/ducks/swaps/swaps.js b/ui/ducks/swaps/swaps.js index c9907b370b2d..a767fd8e5c52 100644 --- a/ui/ducks/swaps/swaps.js +++ b/ui/ducks/swaps/swaps.js @@ -63,7 +63,10 @@ import { checkNetworkAndAccountSupports1559, } from '../../selectors'; -import { MetaMetricsEventCategory } from '../../../shared/constants/metametrics'; +import { + MetaMetricsEventCategory, + MetaMetricsEventName, +} from '../../../shared/constants/metametrics'; import { ERROR_FETCHING_QUOTES, QUOTES_NOT_AVAILABLE_ERROR, @@ -787,6 +790,18 @@ export const fetchQuotesAndSetQuoteState = ( } else { const newSelectedQuote = fetchedQuotes[selectedAggId]; + const tokenToAmountBN = calcTokenAmount( + newSelectedQuote.destinationAmount, + newSelectedQuote.decimals || 18, + ); + + // Firefox and Chrome have different implementations of the APIs + // that we rely on for communication accross the app. On Chrome big + // numbers are converted into number strings, on firefox they remain + // Big Number objects. As such, we convert them here for both + // browsers. + const tokenToAmountToString = tokenToAmountBN.toString(10); + trackEvent({ event: 'Quotes Received', category: MetaMetricsEventCategory.Swaps, @@ -794,10 +809,7 @@ export const fetchQuotesAndSetQuoteState = ( token_from: fromTokenSymbol, token_from_amount: String(inputValue), token_to: toTokenSymbol, - token_to_amount: calcTokenAmount( - newSelectedQuote.destinationAmount, - newSelectedQuote.decimals || 18, - ), + token_to_amount: tokenToAmountToString, request_type: balanceError ? 'Quote' : 'Order', slippage: maxSlippage, custom_slippage: maxSlippage !== Slippage.default, @@ -1177,7 +1189,7 @@ export const signAndSendTransactions = ( } trackEvent({ - event: 'Swap Started', + event: MetaMetricsEventName.SwapStarted, category: MetaMetricsEventCategory.Swaps, sensitiveProperties: swapMetaData, }); From b576c5245ccd2811de5ef9c9953555ea6a78fe99 Mon Sep 17 00:00:00 2001 From: cryptodev-2s <109512101+cryptodev-2s@users.noreply.github.com> Date: Fri, 28 Jul 2023 20:09:14 +0100 Subject: [PATCH 09/25] Use getKeyringForAccount from core KeyringController (#20202) --- .../controllers/encryption-public-key.test.ts | 43 ++++++------------- .../controllers/encryption-public-key.ts | 23 ++++++---- app/scripts/metamask-controller.js | 28 ++++++++---- app/scripts/metamask-controller.test.js | 9 ++-- 4 files changed, 54 insertions(+), 49 deletions(-) diff --git a/app/scripts/controllers/encryption-public-key.test.ts b/app/scripts/controllers/encryption-public-key.test.ts index cb5581831055..c36418abffb1 100644 --- a/app/scripts/controllers/encryption-public-key.test.ts +++ b/app/scripts/controllers/encryption-public-key.test.ts @@ -19,7 +19,7 @@ const messageIdMock2 = '456'; const stateMock = { test: 123 }; const addressMock = '0xc38bf1ad06ef69f0c04e29dbeb4152b4175f0a8d'; const publicKeyMock = '32762347862378feb87123781623a='; -const keyringMock = { type: KeyringType.hdKeyTree }; +const keyringTypeMock = KeyringType.hdKeyTree; const messageParamsMock = { from: addressMock, @@ -73,11 +73,6 @@ const createEncryptionPublicKeyManagerMock = () => }, } as any as jest.Mocked); -const createKeyringControllerMock = () => ({ - getKeyringForAccount: jest.fn(), - getEncryptionPublicKey: jest.fn(), -}); - describe('EncryptionPublicKeyController', () => { let encryptionPublicKeyController: EncryptionPublicKeyController; @@ -88,7 +83,8 @@ describe('EncryptionPublicKeyController', () => { const encryptionPublicKeyManagerMock = createEncryptionPublicKeyManagerMock(); const messengerMock = createMessengerMock(); - const keyringControllerMock = createKeyringControllerMock(); + const getEncryptionPublicKeyMock = jest.fn(); + const getAccountKeyringTypeMock = jest.fn(); const getStateMock = jest.fn(); const metricsEventMock = jest.fn(); @@ -101,7 +97,8 @@ describe('EncryptionPublicKeyController', () => { encryptionPublicKeyController = new EncryptionPublicKeyController({ messenger: messengerMock as any, - keyringController: keyringControllerMock as any, + getEncryptionPublicKey: getEncryptionPublicKeyMock as any, + getAccountKeyringType: getAccountKeyringTypeMock as any, getState: getStateMock as any, metricsEvent: metricsEventMock as any, } as EncryptionPublicKeyControllerOptions); @@ -203,9 +200,7 @@ describe('EncryptionPublicKeyController', () => { ])( 'throws if keyring is not supported', async (keyringName, keyringType) => { - keyringControllerMock.getKeyringForAccount.mockResolvedValueOnce({ - type: keyringType, - }); + getAccountKeyringTypeMock.mockResolvedValueOnce(keyringType); await expect( encryptionPublicKeyController.newRequestEncryptionPublicKey( @@ -219,9 +214,7 @@ describe('EncryptionPublicKeyController', () => { ); it('adds message to message manager', async () => { - keyringControllerMock.getKeyringForAccount.mockResolvedValueOnce( - keyringMock, - ); + getAccountKeyringTypeMock.mockResolvedValueOnce(keyringTypeMock); await encryptionPublicKeyController.newRequestEncryptionPublicKey( addressMock, @@ -243,9 +236,7 @@ describe('EncryptionPublicKeyController', () => { from: messageParamsMock.data, }); - keyringControllerMock.getEncryptionPublicKey.mockResolvedValueOnce( - publicKeyMock, - ); + getEncryptionPublicKeyMock.mockResolvedValueOnce(publicKeyMock); }); it('approves message and signs', async () => { @@ -253,10 +244,8 @@ describe('EncryptionPublicKeyController', () => { messageParamsMock, ); - expect( - keyringControllerMock.getEncryptionPublicKey, - ).toHaveBeenCalledTimes(1); - expect(keyringControllerMock.getEncryptionPublicKey).toHaveBeenCalledWith( + expect(getEncryptionPublicKeyMock).toHaveBeenCalledTimes(1); + expect(getEncryptionPublicKeyMock).toHaveBeenCalledWith( messageParamsMock.data, ); @@ -294,10 +283,8 @@ describe('EncryptionPublicKeyController', () => { }); it('rejects message on error', async () => { - keyringControllerMock.getEncryptionPublicKey.mockReset(); - keyringControllerMock.getEncryptionPublicKey.mockRejectedValue( - new Error('Test Error'), - ); + getEncryptionPublicKeyMock.mockReset(); + getEncryptionPublicKeyMock.mockRejectedValue(new Error('Test Error')); await expect( encryptionPublicKeyController.encryptionPublicKey(messageParamsMock), @@ -312,10 +299,8 @@ describe('EncryptionPublicKeyController', () => { }); it('rejects approval on error', async () => { - keyringControllerMock.getEncryptionPublicKey.mockReset(); - keyringControllerMock.getEncryptionPublicKey.mockRejectedValue( - new Error('Test Error'), - ); + getEncryptionPublicKeyMock.mockReset(); + getEncryptionPublicKeyMock.mockRejectedValue(new Error('Test Error')); await expect( encryptionPublicKeyController.encryptionPublicKey(messageParamsMock), diff --git a/app/scripts/controllers/encryption-public-key.ts b/app/scripts/controllers/encryption-public-key.ts index da50f1afe4aa..8448fba8a8b6 100644 --- a/app/scripts/controllers/encryption-public-key.ts +++ b/app/scripts/controllers/encryption-public-key.ts @@ -4,7 +4,6 @@ import { EncryptionPublicKeyManager, EncryptionPublicKeyParamsMetamask, } from '@metamask/message-manager'; -import { KeyringController } from '@metamask/eth-keyring-controller'; import { AbstractMessageManager, AbstractMessage, @@ -83,7 +82,8 @@ export type EncryptionPublicKeyControllerMessenger = export type EncryptionPublicKeyControllerOptions = { messenger: EncryptionPublicKeyControllerMessenger; - keyringController: KeyringController; + getEncryptionPublicKey: (address: string) => Promise; + getAccountKeyringType: (account: string) => Promise; getState: () => any; metricsEvent: (payload: any, options?: any) => void; }; @@ -98,7 +98,9 @@ export default class EncryptionPublicKeyController extends BaseControllerV2< > { hub: EventEmitter; - private _keyringController: KeyringController; + private _getEncryptionPublicKey: (address: string) => Promise; + + private _getAccountKeyringType: (account: string) => Promise; private _getState: () => any; @@ -111,13 +113,15 @@ export default class EncryptionPublicKeyController extends BaseControllerV2< * * @param options - The controller options. * @param options.messenger - The restricted controller messenger for the EncryptionPublicKey controller. - * @param options.keyringController - An instance of a keyring controller used to extract the encryption public key. + * @param options.getEncryptionPublicKey - Callback to get the keyring encryption public key. + * @param options.getAccountKeyringType - Callback to get the keyring type. * @param options.getState - Callback to retrieve all user state. * @param options.metricsEvent - A function for emitting a metric event. */ constructor({ messenger, - keyringController, + getEncryptionPublicKey, + getAccountKeyringType, getState, metricsEvent, }: EncryptionPublicKeyControllerOptions) { @@ -128,7 +132,8 @@ export default class EncryptionPublicKeyController extends BaseControllerV2< state: getDefaultState(), }); - this._keyringController = keyringController; + this._getEncryptionPublicKey = getEncryptionPublicKey; + this._getAccountKeyringType = getAccountKeyringType; this._getState = getState; this._metricsEvent = metricsEvent; @@ -186,9 +191,9 @@ export default class EncryptionPublicKeyController extends BaseControllerV2< address: string, req: OriginalRequest, ): Promise { - const keyring = await this._keyringController.getKeyringForAccount(address); + const keyringType = await this._getAccountKeyringType(address); - switch (keyring.type) { + switch (keyringType) { case KeyringType.ledger: { return new Promise((_, reject) => { reject( @@ -244,7 +249,7 @@ export default class EncryptionPublicKeyController extends BaseControllerV2< await this._encryptionPublicKeyManager.approveMessage(msgParams); // EncryptionPublicKey message - const publicKey = await this._keyringController.getEncryptionPublicKey( + const publicKey = await this._getEncryptionPublicKey( cleanMessageParams.from, ); diff --git a/app/scripts/metamask-controller.js b/app/scripts/metamask-controller.js index 7ff6b5d263ab..9efbb8447fd2 100644 --- a/app/scripts/metamask-controller.js +++ b/app/scripts/metamask-controller.js @@ -904,9 +904,8 @@ export default class MetamaskController extends EventEmitter { (address) => !identities[address], ); const keyringTypesWithMissingIdentities = - accountsMissingIdentities.map( - (address) => - this.keyringController.getKeyringForAccount(address)?.type, + accountsMissingIdentities.map((address) => + this.coreKeyringController.getAccountKeyringType(address), ); const identitiesCount = Object.keys(identities || {}).length; @@ -1349,7 +1348,14 @@ export default class MetamaskController extends EventEmitter { `${this.approvalController.name}:rejectRequest`, ], }), - keyringController: this.keyringController, + getEncryptionPublicKey: + this.keyringController.getEncryptionPublicKey.bind( + this.keyringController, + ), + getAccountKeyringType: + this.coreKeyringController.getAccountKeyringType.bind( + this.coreKeyringController, + ), getState: this.getState.bind(this), metricsEvent: this.metaMetricsController.trackEvent.bind( this.metaMetricsController, @@ -3265,8 +3271,10 @@ export default class MetamaskController extends EventEmitter { * @returns {'hardware' | 'imported' | 'MetaMask'} */ async getAccountType(address) { - const keyring = await this.keyringController.getKeyringForAccount(address); - switch (keyring.type) { + const keyringType = await this.coreKeyringController.getAccountKeyringType( + address, + ); + switch (keyringType) { case KeyringType.trezor: case KeyringType.lattice: case KeyringType.qr: @@ -3288,7 +3296,9 @@ export default class MetamaskController extends EventEmitter { * @returns {'ledger' | 'lattice' | 'N/A' | string} */ async getDeviceModel(address) { - const keyring = await this.keyringController.getKeyringForAccount(address); + const keyring = await this.coreKeyringController.getKeyringForAccount( + address, + ); switch (keyring.type) { case KeyringType.trezor: return keyring.getModel(); @@ -3527,7 +3537,9 @@ export default class MetamaskController extends EventEmitter { this.custodyController.removeAccount(address); ///: END:ONLY_INCLUDE_IN(build-mmi) - const keyring = await this.keyringController.getKeyringForAccount(address); + const keyring = await this.coreKeyringController.getKeyringForAccount( + address, + ); // Remove account from the keyring await this.keyringController.removeAccount(address); const updatedKeyringAccounts = keyring ? await keyring.getAccounts() : {}; diff --git a/app/scripts/metamask-controller.test.js b/app/scripts/metamask-controller.test.js index 32d7b3c2fbf1..a8d360e10005 100644 --- a/app/scripts/metamask-controller.test.js +++ b/app/scripts/metamask-controller.test.js @@ -859,7 +859,10 @@ describe('MetaMaskController', function () { sinon.stub(metamaskController.keyringController, 'removeAccount'); sinon.stub(metamaskController, 'removeAllAccountPermissions'); sinon - .stub(metamaskController.keyringController, 'getKeyringForAccount') + .stub( + metamaskController.coreKeyringController, + 'getKeyringForAccount', + ) .returns(Promise.resolve(mockKeyring)); ret = await metamaskController.removeAccount(addressToRemove); @@ -906,9 +909,9 @@ describe('MetaMaskController', function () { it('should return address', async function () { assert.equal(ret, '0x1'); }); - it('should call keyringController.getKeyringForAccount', async function () { + it('should call coreKeyringController.getKeyringForAccount', async function () { assert( - metamaskController.keyringController.getKeyringForAccount.calledWith( + metamaskController.coreKeyringController.getKeyringForAccount.calledWith( addressToRemove, ), ); From 6546aad3345592cd073ebcd8d33d81c983263fd7 Mon Sep 17 00:00:00 2001 From: jainex Date: Sat, 29 Jul 2023 02:33:27 +0530 Subject: [PATCH 10/25] Replacing deprecated Popover with Modal (#19784) * Replacing deprecated Popover with Modal * Fix Jest error --- .../cancel-speedup-popover.js | 133 ++++++++++-------- 1 file changed, 73 insertions(+), 60 deletions(-) diff --git a/ui/components/app/cancel-speedup-popover/cancel-speedup-popover.js b/ui/components/app/cancel-speedup-popover/cancel-speedup-popover.js index a1eda061c878..f244f3f63701 100644 --- a/ui/components/app/cancel-speedup-popover/cancel-speedup-popover.js +++ b/ui/components/app/cancel-speedup-popover/cancel-speedup-popover.js @@ -16,9 +16,16 @@ import EditGasFeeButton from '../edit-gas-fee-button'; import GasDetailsItem from '../gas-details-item'; import Box from '../../ui/box'; import InfoTooltip from '../../ui/info-tooltip'; -import Popover from '../../ui/popover'; import AppLoadingSpinner from '../app-loading-spinner'; -import { Button, ButtonLink, Text } from '../../component-library'; +import { + Text, + Button, + ButtonLink, + Modal, + ModalOverlay, + ModalContent, + ModalHeader, +} from '../../component-library'; const CancelSpeedupPopover = () => { const { @@ -82,68 +89,74 @@ const CancelSpeedupPopover = () => { }; return ( - - {editGasMode === EditGasModes.cancel - ? `❌${t('cancel')}` - : `🚀${t('speedUp')}`} - - } + closeModal(['cancelSpeedUpTransaction'])} className="cancel-speedup-popover" > - -
- - {t('cancelSpeedUpLabel', [ - {t('replace')}, - ])} - - - {t('cancelSpeedUpTransactionTooltip', [ - editGasMode === EditGasModes.cancel - ? t('cancel') - : t('speedUp'), - ])} - - - {t('learnMoreUpperCase')} - - - } - /> - - + + closeModal(['cancelSpeedUpTransaction'])} + marginBottom={4} > -
- {!appIsLoading && } -
-
- -
-
- -
-
+ {editGasMode === EditGasModes.cancel + ? `❌${t('cancel')}` + : `🚀${t('speedUp')}`} + + + +
+ + {t('cancelSpeedUpLabel', [ + {t('replace')}, + ])} + + + {t('cancelSpeedUpTransactionTooltip', [ + editGasMode === EditGasModes.cancel + ? t('cancel') + : t('speedUp'), + ])} + + + {t('learnMoreUpperCase')} + + + } + /> + + +
+ {!appIsLoading && } +
+
+ +
+
+ +
+ + ); }; From 0129ea913f8e27102a695ca8b399ded9e2f63668 Mon Sep 17 00:00:00 2001 From: David Walsh Date: Fri, 28 Jul 2023 18:18:22 -0500 Subject: [PATCH 11/25] Fix #20162 - Add Whats New for Global Menu (#20244) --- app/_locales/en/messages.json | 9 + app/images/global-menu-block-explorer.svg | 827 ++++++++++++++++++++++ shared/notifications/index.js | 19 + test/e2e/fixture-builder.js | 5 + 4 files changed, 860 insertions(+) create mode 100644 app/images/global-menu-block-explorer.svg diff --git a/app/_locales/en/messages.json b/app/_locales/en/messages.json index fdd3a614d3e8..99d3c8a92524 100644 --- a/app/_locales/en/messages.json +++ b/app/_locales/en/messages.json @@ -2737,6 +2737,15 @@ "notifications21Title": { "message": "Introducing new and refreshed Swaps!" }, + "notifications22ActionText": { + "message": "Got it" + }, + "notifications22Description": { + "message": "💡 Just click the global menu or account menu to find them!" + }, + "notifications22Title": { + "message": "Looking for your account details or the block explorer URL?" + }, "notifications3ActionText": { "message": "Read more", "description": "The 'call to action' on the button, or link, of the 'Stay secure' notification. Upon clicking, users will be taken to a page about security on the metamask support website." diff --git a/app/images/global-menu-block-explorer.svg b/app/images/global-menu-block-explorer.svg new file mode 100644 index 000000000000..728459030838 --- /dev/null +++ b/app/images/global-menu-block-explorer.svg @@ -0,0 +1,827 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/shared/notifications/index.js b/shared/notifications/index.js index a5a689f3a3c5..bc74e3222039 100644 --- a/shared/notifications/index.js +++ b/shared/notifications/index.js @@ -114,6 +114,14 @@ export const UI_NOTIFICATIONS = { width: '100%', }, }, + 22: { + id: 22, + date: null, + image: { + src: 'images/global-menu-block-explorer.svg', + width: '100%', + }, + }, }; export const getTranslatedUINotifications = (t, locale) => { @@ -313,5 +321,16 @@ export const getTranslatedUINotifications = (t, locale) => { ) : '', }, + 22: { + ...UI_NOTIFICATIONS[22], + title: t('notifications22Title'), + description: t('notifications22Description'), + actionText: t('notifications22ActionText'), + date: UI_NOTIFICATIONS[22].date + ? new Intl.DateTimeFormat(formattedLocale).format( + new Date(UI_NOTIFICATIONS[22].date), + ) + : '', + }, }; }; diff --git a/test/e2e/fixture-builder.js b/test/e2e/fixture-builder.js index bbbcd49cf42b..ed3e3aaacb5e 100644 --- a/test/e2e/fixture-builder.js +++ b/test/e2e/fixture-builder.js @@ -141,6 +141,11 @@ function defaultFixture() { id: 21, isShown: true, }, + 22: { + date: null, + id: 22, + isShown: true, + }, }, }, AppStateController: { From 96e4b7bb9f46f8d9819c6294d731b876210215da Mon Sep 17 00:00:00 2001 From: Harsh Shukla <125105825+PrgrmrHarshShukla@users.noreply.github.com> Date: Sat, 29 Jul 2023 18:55:34 +0530 Subject: [PATCH 12/25] Replacing deprecated code in favor of component-library components (#20137) --- .../app/snaps/snap-content-footer/index.scss | 7 +--- .../snap-content-footer.js | 37 ++++++++++++++----- .../snap-content-footer.stories.js | 2 +- 3 files changed, 30 insertions(+), 16 deletions(-) diff --git a/ui/components/app/snaps/snap-content-footer/index.scss b/ui/components/app/snaps/snap-content-footer/index.scss index f337b26ad43f..b9a6e3cf2535 100644 --- a/ui/components/app/snaps/snap-content-footer/index.scss +++ b/ui/components/app/snaps/snap-content-footer/index.scss @@ -1,8 +1,5 @@ .snap-content-footer { - .button { - white-space: nowrap; - overflow: hidden; - text-overflow: ellipsis; - width: 100px; + &__description { + max-width: 100%; // helps with snap name truncation } } diff --git a/ui/components/app/snaps/snap-content-footer/snap-content-footer.js b/ui/components/app/snaps/snap-content-footer/snap-content-footer.js index f8b31826aebc..c0c90e8fb2bb 100644 --- a/ui/components/app/snaps/snap-content-footer/snap-content-footer.js +++ b/ui/components/app/snaps/snap-content-footer/snap-content-footer.js @@ -3,20 +3,26 @@ import PropTypes from 'prop-types'; import { useHistory } from 'react-router-dom'; -import Typography from '../../../ui/typography/typography'; import { useI18nContext } from '../../../../hooks/useI18nContext'; import { SNAPS_VIEW_ROUTE } from '../../../../helpers/constants/routes'; import { - TypographyVariant, + TextVariant, JustifyContent, AlignItems, TextColor, Size, IconColor, + Display, } from '../../../../helpers/constants/design-system'; -import Button from '../../../ui/button'; -import Box from '../../../ui/box/box'; -import { Icon, IconName } from '../../../component-library'; +import { + BUTTON_SIZES, + BUTTON_VARIANT, + Box, + Button, + Icon, + IconName, + Text, +} from '../../../component-library'; export default function SnapContentFooter({ snapName, snapId }) { const t = useI18nContext(); @@ -26,9 +32,10 @@ export default function SnapContentFooter({ snapName, snapId }) { e.stopPropagation(); history.push(`${SNAPS_VIEW_ROUTE}/${encodeURIComponent(snapId)}`); }; - // TODO: add truncation to the snap name, need to pick a character length at which to cut off + return ( - + {t('snapContent', [ - , ])} - + ); } diff --git a/ui/components/app/snaps/snap-content-footer/snap-content-footer.stories.js b/ui/components/app/snaps/snap-content-footer/snap-content-footer.stories.js index 95ef4779703b..2d789f2c137d 100644 --- a/ui/components/app/snaps/snap-content-footer/snap-content-footer.stories.js +++ b/ui/components/app/snaps/snap-content-footer/snap-content-footer.stories.js @@ -7,7 +7,7 @@ export default { component: SnapContentFooter, args: { - snapName: 'Test Snap', + snapName: 'Really Long Test Snap Name', snapId: 'local:test-snap', }, }; From e6a4c63d51858d5327b94c7af683a72f9081fb73 Mon Sep 17 00:00:00 2001 From: David Walsh Date: Sat, 29 Jul 2023 14:59:24 -0500 Subject: [PATCH 13/25] Fix #19950 - Use removable property to determine if network should be removable (#20220) --- .../network-list-menu/network-list-menu.js | 12 ++------ ui/selectors/selectors.js | 17 +++++++---- ui/selectors/selectors.test.js | 28 +++++++++++++++++++ 3 files changed, 41 insertions(+), 16 deletions(-) diff --git a/ui/components/multichain/network-list-menu/network-list-menu.js b/ui/components/multichain/network-list-menu/network-list-menu.js index 9fd49b9bdb4f..baf7e851e3d8 100644 --- a/ui/components/multichain/network-list-menu/network-list-menu.js +++ b/ui/components/multichain/network-list-menu/network-list-menu.js @@ -12,7 +12,7 @@ import { setProviderType, toggleNetworkMenu, } from '../../../store/actions'; -import { CHAIN_IDS, TEST_CHAINS } from '../../../../shared/constants/network'; +import { TEST_CHAINS } from '../../../../shared/constants/network'; import { getShowTestNetworks, getCurrentChainId, @@ -52,12 +52,6 @@ import { isLineaMainnetNetworkReleased, } from '../../../ducks/metamask/metamask'; -const UNREMOVABLE_CHAIN_IDS = [ - CHAIN_IDS.MAINNET, - CHAIN_IDS.LINEA_MAINNET, - ...TEST_CHAINS, -]; - export const NetworkListMenu = ({ onClose }) => { const t = useI18nContext(); @@ -117,9 +111,7 @@ export const NetworkListMenu = ({ onClose }) => { } const isCurrentNetwork = currentNetwork.id === network.id; - - const canDeleteNetwork = - !isCurrentNetwork && !UNREMOVABLE_CHAIN_IDS.includes(network.chainId); + const canDeleteNetwork = !isCurrentNetwork && network.removable; return ( chainId === CHAIN_IDS.LOCALHOST, - ), + ...Object.values(networkConfigurations) + .filter(({ chainId }) => chainId === CHAIN_IDS.LOCALHOST) + .map((network) => ({ ...network, removable: true })), ]; } @@ -1253,6 +1256,7 @@ export function getNonTestNetworks(state) { providerType: NETWORK_TYPES.MAINNET, ticker: CURRENCY_SYMBOLS.ETH, id: NETWORK_TYPES.MAINNET, + removable: false, }, { chainId: CHAIN_IDS.LINEA_MAINNET, @@ -1264,11 +1268,12 @@ export function getNonTestNetworks(state) { providerType: NETWORK_TYPES.LINEA_MAINNET, ticker: TEST_NETWORK_TICKER_MAP[NETWORK_TYPES.LINEA_MAINNET], id: NETWORK_TYPES.LINEA_MAINNET, + removable: false, }, // Custom networks added by the user - ...Object.values(networkConfigurations).filter( - ({ chainId }) => ![CHAIN_IDS.LOCALHOST].includes(chainId), - ), + ...Object.values(networkConfigurations) + .filter(({ chainId }) => ![CHAIN_IDS.LOCALHOST].includes(chainId)) + .map((network) => ({ ...network, removable: true })), ]; } diff --git a/ui/selectors/selectors.test.js b/ui/selectors/selectors.test.js index e63252ee70e7..307cbde925f0 100644 --- a/ui/selectors/selectors.test.js +++ b/ui/selectors/selectors.test.js @@ -4,6 +4,7 @@ import { KeyringType } from '../../shared/constants/keyring'; import { CHAIN_IDS, LOCALHOST_DISPLAY_NAME, + NETWORK_TYPES, } from '../../shared/constants/network'; import * as selectors from './selectors'; @@ -314,6 +315,33 @@ describe('Selectors', () => { const lastItem = networks.pop(); expect(lastItem.nickname.toLowerCase()).toContain('localhost'); }); + + it('properly assigns a network as removable', () => { + const networks = selectors.getAllNetworks({ + metamask: { + preferences: { + showTestNetworks: true, + }, + networkConfigurations: { + 'some-config-name': { + chainId: CHAIN_IDS.LOCALHOST, + nickname: LOCALHOST_DISPLAY_NAME, + id: 'some-config-name', + }, + }, + }, + }); + + const mainnet = networks.find( + (network) => network.id === NETWORK_TYPES.MAINNET, + ); + expect(mainnet.removable).toBe(false); + + const customNetwork = networks.find( + (network) => network.id === 'some-config-name', + ); + expect(customNetwork.removable).toBe(true); + }); }); describe('#getCurrentNetwork', () => { From 252c7c1d6d3e7625e2ca7c09d1f1d70b844bb5e7 Mon Sep 17 00:00:00 2001 From: Vinicius Stevam <45455812+vinistevam@users.noreply.github.com> Date: Mon, 31 Jul 2023 11:06:18 +0100 Subject: [PATCH 14/25] Remove unnecessary log from accept approval (#20233) --- app/scripts/controllers/app-state.js | 3 +- app/scripts/controllers/app-state.test.js | 49 ----------------------- 2 files changed, 1 insertion(+), 51 deletions(-) diff --git a/app/scripts/controllers/app-state.js b/app/scripts/controllers/app-state.js index 1bbf4a4f8dc5..cd477fd8d817 100644 --- a/app/scripts/controllers/app-state.js +++ b/app/scripts/controllers/app-state.js @@ -452,7 +452,6 @@ export default class AppStateController extends EventEmitter { _acceptApproval() { if (!this._approvalRequestId) { - log.error('Attempted to accept missing unlock approval request'); return; } try { @@ -461,7 +460,7 @@ export default class AppStateController extends EventEmitter { this._approvalRequestId, ); } catch (error) { - log.error('Failed to accept transaction approval request', error); + log.error('Failed to unlock approval request', error); } this._approvalRequestId = null; diff --git a/app/scripts/controllers/app-state.test.js b/app/scripts/controllers/app-state.test.js index b96d39a4c41d..4716e30cd3d7 100644 --- a/app/scripts/controllers/app-state.test.js +++ b/app/scripts/controllers/app-state.test.js @@ -1,10 +1,7 @@ import { ObservableStore } from '@metamask/obs-store'; -import log from 'loglevel'; import { ORIGIN_METAMASK } from '../../../shared/constants/app'; import AppStateController from './app-state'; -jest.mock('loglevel'); - let appStateController, mockStore; describe('AppStateController', () => { @@ -147,52 +144,6 @@ describe('AppStateController', () => { expect.any(String), ); }); - - it('logs if rejecting approval request throws', async () => { - appStateController._approvalRequestId = 'mock-approval-request-id'; - appStateController = new AppStateController({ - addUnlockListener: jest.fn(), - isUnlocked: jest.fn(() => true), - onInactiveTimeout: jest.fn(), - showUnlockRequest: jest.fn(), - preferencesStore: { - subscribe: jest.fn(), - getState: jest.fn(() => ({ - preferences: { - autoLockTimeLimit: 0, - }, - })), - }, - qrHardwareStore: { - subscribe: jest.fn(), - }, - messenger: { - call: jest.fn(() => { - throw new Error('mock error'); - }), - }, - }); - - appStateController.handleUnlock(); - - expect(log.error).toHaveBeenCalledTimes(1); - expect(log.error).toHaveBeenCalledWith( - 'Attempted to accept missing unlock approval request', - ); - }); - - it('returns without call messenger if no approval request in pending', async () => { - const emitSpy = jest.spyOn(appStateController, 'emit'); - - appStateController.handleUnlock(); - - expect(emitSpy).toHaveBeenCalledTimes(0); - expect(appStateController.messagingSystem.call).toHaveBeenCalledTimes(0); - expect(log.error).toHaveBeenCalledTimes(1); - expect(log.error).toHaveBeenCalledWith( - 'Attempted to accept missing unlock approval request', - ); - }); }); describe('setDefaultHomeActiveTabName', () => { From b76875ac69f52f5b864c42a0c2842b269615ee51 Mon Sep 17 00:00:00 2001 From: Nicholas Ellul Date: Mon, 31 Jul 2023 08:48:48 -0400 Subject: [PATCH 15/25] Update @metamask/phishing-controller to v4.0.0 (#18840) * Update phishing controller to v4.0.0 * Move phishing e2e test utilities into its own helper.js * Update phishing detection e2e test * Update MetaMask Controller test mocks * Update mv3 phishing tests * Fix test for 500 error on warning page * Allow for directories in test folder * Update migration number * Linting fixes * Remove fail on console error * Separate mocks from helpers * Have migration delete PhishingController state entirely * Remove phishing detection directory * Only delete the listState in migration * Bump migration version --- .../metamask-controller.actions.test.js | 44 ++++- app/scripts/metamask-controller.test.js | 43 ++++- app/scripts/migrations/090.test.js | 109 +++++++++++ app/scripts/migrations/090.ts | 37 ++++ app/scripts/migrations/index.js | 2 + package.json | 2 +- test/e2e/helpers.js | 81 --------- test/e2e/mock-e2e.js | 32 +--- .../mv3/phishing-warning-sw-restart.spec.js | 17 +- test/e2e/run-all.js | 18 +- test/e2e/tests/phishing-controller/helpers.js | 25 +++ test/e2e/tests/phishing-controller/mocks.js | 172 ++++++++++++++++++ .../phishing-detection.spec.js | 117 ++++++++---- yarn.lock | 14 +- 14 files changed, 542 insertions(+), 171 deletions(-) create mode 100644 app/scripts/migrations/090.test.js create mode 100644 app/scripts/migrations/090.ts create mode 100644 test/e2e/tests/phishing-controller/helpers.js create mode 100644 test/e2e/tests/phishing-controller/mocks.js rename test/e2e/tests/{ => phishing-controller}/phishing-detection.spec.js (69%) diff --git a/app/scripts/metamask-controller.actions.test.js b/app/scripts/metamask-controller.actions.test.js index c1ea740f81fb..5e7c771a1a6b 100644 --- a/app/scripts/metamask-controller.actions.test.js +++ b/app/scripts/metamask-controller.actions.test.js @@ -1,7 +1,14 @@ import { strict as assert } from 'assert'; import sinon from 'sinon'; import proxyquire from 'proxyquire'; - +import { + ListNames, + METAMASK_STALELIST_URL, + METAMASK_HOTLIST_DIFF_URL, + PHISHING_CONFIG_BASE_URL, + METAMASK_STALELIST_FILE, + METAMASK_HOTLIST_DIFF_FILE, +} from '@metamask/phishing-controller'; import { ApprovalRequestNotFoundError } from '@metamask/approval-controller'; import { PermissionsRequestNotFoundError } from '@metamask/permission-controller'; import nock from 'nock'; @@ -59,21 +66,28 @@ describe('MetaMaskController', function () { }); beforeEach(function () { - nock('https://static.metafi.codefi.network') + nock(PHISHING_CONFIG_BASE_URL) .persist() - .get('/api/v1/lists/stalelist.json') + .get(METAMASK_STALELIST_FILE) .reply( 200, JSON.stringify({ version: 2, tolerance: 2, - fuzzylist: [], - allowlist: [], - blocklist: ['127.0.0.1'], - lastUpdated: 0, + lastUpdated: 1, + eth_phishing_detect_config: { + fuzzylist: [], + allowlist: [], + blocklist: ['127.0.0.1'], + name: ListNames.MetaMask, + }, + phishfort_hotlist: { + blocklist: [], + name: ListNames.Phishfort, + }, }), ) - .get('/api/v1/lists/hotlist.json') + .get(METAMASK_HOTLIST_DIFF_FILE) .reply( 200, JSON.stringify([ @@ -110,6 +124,20 @@ describe('MetaMaskController', function () { await ganacheServer.quit(); }); + describe('Phishing Detection Mock', function () { + it('should be updated to use v1 of the API', function () { + // Update the fixture above if this test fails + assert.equal( + METAMASK_STALELIST_URL, + 'https://phishing-detection.metafi.codefi.network/v1/stalelist', + ); + assert.equal( + METAMASK_HOTLIST_DIFF_URL, + 'https://phishing-detection.metafi.codefi.network/v1/diffsSince', + ); + }); + }); + describe('#addNewAccount', function () { it('two parallel calls with same accountCount give same result', async function () { await metamaskController.createNewVaultAndKeychain('test@123'); diff --git a/app/scripts/metamask-controller.test.js b/app/scripts/metamask-controller.test.js index a8d360e10005..5bed8c899d51 100644 --- a/app/scripts/metamask-controller.test.js +++ b/app/scripts/metamask-controller.test.js @@ -7,6 +7,14 @@ import EthQuery from 'eth-query'; import proxyquire from 'proxyquire'; import browser from 'webextension-polyfill'; import { wordlist as englishWordlist } from '@metamask/scure-bip39/dist/wordlists/english'; +import { + ListNames, + METAMASK_STALELIST_URL, + METAMASK_HOTLIST_DIFF_URL, + PHISHING_CONFIG_BASE_URL, + METAMASK_STALELIST_FILE, + METAMASK_HOTLIST_DIFF_FILE, +} from '@metamask/phishing-controller'; import { TransactionStatus } from '../../shared/constants/transaction'; import createTxMeta from '../../test/lib/createTxMeta'; import { NETWORK_TYPES } from '../../shared/constants/network'; @@ -185,21 +193,28 @@ describe('MetaMaskController', function () { .persist() .get(/.*/u) .reply(200, '{"JPY":12415.9}'); - nock('https://static.metafi.codefi.network') + nock(PHISHING_CONFIG_BASE_URL) .persist() - .get('/api/v1/lists/stalelist.json') + .get(METAMASK_STALELIST_FILE) .reply( 200, JSON.stringify({ version: 2, tolerance: 2, - fuzzylist: [], - allowlist: [], - blocklist: ['127.0.0.1'], - lastUpdated: 0, + lastUpdated: 1, + eth_phishing_detect_config: { + fuzzylist: [], + allowlist: [], + blocklist: ['127.0.0.1'], + name: ListNames.MetaMask, + }, + phishfort_hotlist: { + blocklist: [], + name: ListNames.Phishfort, + }, }), ) - .get('/api/v1/lists/hotlist.json') + .get(METAMASK_HOTLIST_DIFF_FILE) .reply( 200, JSON.stringify([ @@ -223,6 +238,20 @@ describe('MetaMaskController', function () { await ganacheServer.quit(); }); + describe('Phishing Detection Mock', function () { + it('should be updated to use v1 of the API', function () { + // Update the fixture above if this test fails + assert.equal( + METAMASK_STALELIST_URL, + 'https://phishing-detection.metafi.codefi.network/v1/stalelist', + ); + assert.equal( + METAMASK_HOTLIST_DIFF_URL, + 'https://phishing-detection.metafi.codefi.network/v1/diffsSince', + ); + }); + }); + describe('MetaMaskController Behaviour', function () { let metamaskController; diff --git a/app/scripts/migrations/090.test.js b/app/scripts/migrations/090.test.js new file mode 100644 index 000000000000..6a28c60f2d76 --- /dev/null +++ b/app/scripts/migrations/090.test.js @@ -0,0 +1,109 @@ +import { migrate, version } from './090'; + +const PREVIOUS_VERSION = version - 1; + +describe('migration #90', () => { + it('updates the version metadata', async () => { + const oldStorage = { + meta: { + version: PREVIOUS_VERSION, + }, + data: {}, + }; + + const newStorage = await migrate(oldStorage); + + expect(newStorage.meta).toStrictEqual({ + version, + }); + }); + + it('does not change the state if the phishing controller state does not exist', async () => { + const oldStorage = { + meta: { + version: PREVIOUS_VERSION, + }, + data: { test: '123' }, + }; + + const newStorage = await migrate(oldStorage); + + expect(newStorage.data).toStrictEqual(oldStorage.data); + }); + + it('does not change the state if the phishing controller state is invalid', async () => { + const oldStorage = { + meta: { + version: PREVIOUS_VERSION, + }, + data: { PhishingController: 'this is not valid' }, + }; + + const newStorage = await migrate(oldStorage); + + expect(newStorage.data).toStrictEqual(oldStorage.data); + }); + + it('does not change the state if the listState property does not exist', async () => { + const oldStorage = { + meta: { + version: PREVIOUS_VERSION, + }, + data: { + PhishingController: { test: 123 }, + }, + }; + + const newStorage = await migrate(oldStorage); + + expect(newStorage.data).toStrictEqual(oldStorage.data); + }); + + it('deletes the "listState" property', async () => { + const oldStorage = { + meta: { + version: PREVIOUS_VERSION, + }, + data: { PhishingController: { listState: {} } }, + }; + + const newStorage = await migrate(oldStorage); + + expect(newStorage.data.PhishingController.listState).toBeUndefined(); + }); + + it('deletes the listState if present', async () => { + const oldStorage = { + meta: { + version: PREVIOUS_VERSION, + }, + data: { PhishingController: { listState: { test: 123 } } }, + }; + + const newStorage = await migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + PhishingController: {}, + }); + }); + + it('does not delete the allowlist if present', async () => { + const oldStorage = { + meta: { + version: PREVIOUS_VERSION, + }, + data: { + PhishingController: { + whitelist: ['foobar.com'], + listState: { test: 123 }, + }, + }, + }; + + const newStorage = await migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + PhishingController: { whitelist: ['foobar.com'] }, + }); + }); +}); diff --git a/app/scripts/migrations/090.ts b/app/scripts/migrations/090.ts new file mode 100644 index 000000000000..e45ec05e4168 --- /dev/null +++ b/app/scripts/migrations/090.ts @@ -0,0 +1,37 @@ +import { cloneDeep } from 'lodash'; +import { hasProperty, isObject } from '@metamask/utils'; + +export const version = 90; + +/** + * Explain the purpose of the migration here. + * + * @param originalVersionedData - Versioned MetaMask extension state, exactly what we persist to dist. + * @param originalVersionedData.meta - State metadata. + * @param originalVersionedData.meta.version - The current state version. + * @param originalVersionedData.data - The persisted MetaMask state, keyed by controller. + * @returns Updated versioned MetaMask extension state. + */ +export async function migrate(originalVersionedData: { + meta: { version: number }; + data: Record; +}) { + const versionedData = cloneDeep(originalVersionedData); + versionedData.meta.version = version; + versionedData.data = transformState(versionedData.data); + return versionedData; +} + +function transformState(state: Record) { + if ( + !hasProperty(state, 'PhishingController') || + !isObject(state.PhishingController) || + !hasProperty(state.PhishingController, 'listState') + ) { + return state; + } + + delete state.PhishingController.listState; + + return state; +} diff --git a/app/scripts/migrations/index.js b/app/scripts/migrations/index.js index 429ef7959089..adbe48252599 100644 --- a/app/scripts/migrations/index.js +++ b/app/scripts/migrations/index.js @@ -93,6 +93,7 @@ import * as m086 from './086'; import * as m087 from './087'; import * as m088 from './088'; import * as m089 from './089'; +import * as m090 from './090'; const migrations = [ m002, @@ -183,6 +184,7 @@ const migrations = [ m087, m088, m089, + m090, ]; export default migrations; diff --git a/package.json b/package.json index ca3fada5f362..0261e26af05e 100644 --- a/package.json +++ b/package.json @@ -256,7 +256,7 @@ "@metamask/notification-controller": "^3.0.0", "@metamask/obs-store": "^8.1.0", "@metamask/permission-controller": "^4.0.0", - "@metamask/phishing-controller": "^3.0.0", + "@metamask/phishing-controller": "^4.0.0", "@metamask/post-message-stream": "^6.0.0", "@metamask/ppom-validator": "^0.1.2", "@metamask/providers": "^11.1.0", diff --git a/test/e2e/helpers.js b/test/e2e/helpers.js index 536c13bbb420..cbdacf71f9f3 100644 --- a/test/e2e/helpers.js +++ b/test/e2e/helpers.js @@ -506,85 +506,6 @@ const openDapp = async (driver, contract = null, dappURL = DAPP_URL) => { ? await driver.openNewPage(`${dappURL}/?contract=${contract}`) : await driver.openNewPage(dappURL); }; -const STALELIST_URL = - 'https://static.metafi.codefi.network/api/v1/lists/stalelist.json'; - -const emptyHtmlPage = ` - - - - title - - - Empty page - -`; - -/** - * Setup fetch mocks for the phishing detection feature. - * - * The mock configuration will show that "127.0.0.1" is blocked. The dynamic lookup on the warning - * page can be customized, so that we can test both the MetaMask and PhishFort block cases. - * - * @param {import('mockttp').Mockttp} mockServer - The mock server. - * @param {object} metamaskPhishingConfigResponse - The response for the dynamic phishing - * configuration lookup performed by the warning page. - */ -async function setupPhishingDetectionMocks( - mockServer, - metamaskPhishingConfigResponse, -) { - await mockServer.forGet(STALELIST_URL).thenCallback(() => { - return { - statusCode: 200, - json: { - version: 2, - tolerance: 2, - fuzzylist: [], - allowlist: [], - blocklist: ['127.0.0.1'], - lastUpdated: 0, - }, - }; - }); - - await mockServer - .forGet('/~https://github.com/MetaMask/eth-phishing-detect/issues/new') - .thenCallback(() => { - return { - statusCode: 200, - body: emptyHtmlPage, - }; - }); - await mockServer - .forGet('/~https://github.com/phishfort/phishfort-lists/issues/new') - .thenCallback(() => { - return { - statusCode: 200, - body: emptyHtmlPage, - }; - }); - - await mockServer - .forGet( - 'https://raw.githubusercontent.com/MetaMask/eth-phishing-detect/master/src/config.json', - ) - .thenCallback(() => metamaskPhishingConfigResponse); -} - -function mockPhishingDetection(mockServer) { - setupPhishingDetectionMocks(mockServer, { - statusCode: 200, - json: { - version: 2, - tolerance: 2, - fuzzylist: [], - whitelist: [], - blacklist: ['127.0.0.1'], - lastUpdated: 0, - }, - }); -} const PRIVATE_KEY = '0x7C9529A67102755B7E6102D6D950AC5D5863C98713805CEC576B945B15B71EAC'; @@ -840,8 +761,6 @@ module.exports = { importWrongSRPOnboardingFlow, testSRPDropdownIterations, openDapp, - mockPhishingDetection, - setupPhishingDetectionMocks, defaultGanacheOptions, sendTransaction, findAnotherAccountFromAccountList, diff --git a/test/e2e/mock-e2e.js b/test/e2e/mock-e2e.js index a3e48208dbb7..997c040c290a 100644 --- a/test/e2e/mock-e2e.js +++ b/test/e2e/mock-e2e.js @@ -4,21 +4,9 @@ const blacklistedHosts = [ 'mainnet.infura.io', 'sepolia.infura.io', ]; - -const HOTLIST_URL = - 'https://static.metafi.codefi.network/api/v1/lists/hotlist.json'; -const STALELIST_URL = - 'https://static.metafi.codefi.network/api/v1/lists/stalelist.json'; - -const emptyHotlist = []; -const emptyStalelist = { - version: 2, - tolerance: 2, - fuzzylist: [], - allowlist: [], - blocklist: [], - lastUpdated: 0, -}; +const { + mockEmptyStalelistAndHotlist, +} = require('./tests/phishing-controller/mocks'); /** * Setup E2E network mocks. @@ -385,19 +373,7 @@ async function setupMocking(server, testSpecificMock, { chainId }) { }; }); - await server.forGet(STALELIST_URL).thenCallback(() => { - return { - statusCode: 200, - json: emptyStalelist, - }; - }); - - await server.forGet(HOTLIST_URL).thenCallback(() => { - return { - statusCode: 200, - json: emptyHotlist, - }; - }); + await mockEmptyStalelistAndHotlist(server); await server .forPost('https://customnetwork.com/api/customRPC') diff --git a/test/e2e/mv3/phishing-warning-sw-restart.spec.js b/test/e2e/mv3/phishing-warning-sw-restart.spec.js index 2e10f35b2053..8de3ad7e0c60 100644 --- a/test/e2e/mv3/phishing-warning-sw-restart.spec.js +++ b/test/e2e/mv3/phishing-warning-sw-restart.spec.js @@ -1,7 +1,7 @@ const { strict: assert } = require('assert'); +const FixtureBuilder = require('../fixture-builder'); const { withFixtures, - mockPhishingDetection, openDapp, defaultGanacheOptions, assertAccountBalanceForDOM, @@ -11,7 +11,11 @@ const { unlockWallet, terminateServiceWorker, } = require('../helpers'); -const FixtureBuilder = require('../fixture-builder'); + +const { + setupPhishingDetectionMocks, + BlockProvider, +} = require('../tests/phishing-controller/helpers'); describe('Phishing warning page', function () { const driverOptions = { openDevToolsForTabs: true }; @@ -21,12 +25,17 @@ describe('Phishing warning page', function () { await withFixtures( { - dapp: true, fixtures: new FixtureBuilder().build(), ganacheOptions: defaultGanacheOptions, title: this.test.title, - testSpecificMock: mockPhishingDetection, driverOptions, + testSpecificMock: async (mockServer) => { + return setupPhishingDetectionMocks(mockServer, { + blockProvider: BlockProvider.MetaMask, + blocklist: ['127.0.0.1'], + }); + }, + dapp: true, }, async ({ driver, ganacheServer }) => { await driver.navigate(); diff --git a/test/e2e/run-all.js b/test/e2e/run-all.js index d97d6c8a19cc..28c43da282a2 100644 --- a/test/e2e/run-all.js +++ b/test/e2e/run-all.js @@ -6,10 +6,20 @@ const { runInShell } = require('../../development/lib/run-command'); const { exitWithError } = require('../../development/lib/exit-with-error'); const getTestPathsForTestDir = async (testDir) => { - const testFilenames = await fs.readdir(testDir); - const testPaths = testFilenames.map((filename) => - path.join(testDir, filename), - ); + const testFilenames = await fs.readdir(testDir, { withFileTypes: true }); + const testPaths = []; + + for (const itemInDirectory of testFilenames) { + const fullPath = path.join(testDir, itemInDirectory.name); + + if (itemInDirectory.isDirectory()) { + const subDirPaths = await getTestPathsForTestDir(fullPath); + testPaths.push(...subDirPaths); + } else if (fullPath.endsWith('.spec.js')) { + testPaths.push(fullPath); + } + } + return testPaths; }; diff --git a/test/e2e/tests/phishing-controller/helpers.js b/test/e2e/tests/phishing-controller/helpers.js new file mode 100644 index 000000000000..a00d5ddb398f --- /dev/null +++ b/test/e2e/tests/phishing-controller/helpers.js @@ -0,0 +1,25 @@ +const { + METAMASK_STALELIST_URL, + METAMASK_HOTLIST_DIFF_URL, + ListNames, +} = require('@metamask/phishing-controller'); + +/** + * The block provider names. + * + * @enum {BlockProvider} + * @readonly + * @property {string} MetaMask - The name of the MetaMask block provider. + * @property {string} PhishFort - The name of the PhishFort block provider. + */ +const BlockProvider = { + MetaMask: 'metamask', + PhishFort: 'phishfort', +}; + +module.exports = { + METAMASK_HOTLIST_DIFF_URL, + METAMASK_STALELIST_URL, + BlockProvider, + ListNames, +}; diff --git a/test/e2e/tests/phishing-controller/mocks.js b/test/e2e/tests/phishing-controller/mocks.js new file mode 100644 index 000000000000..f15e4b8487e2 --- /dev/null +++ b/test/e2e/tests/phishing-controller/mocks.js @@ -0,0 +1,172 @@ +const { + METAMASK_STALELIST_URL, + METAMASK_HOTLIST_DIFF_URL, + ListNames, + BlockProvider, +} = require('./helpers'); + +// last updated must not be 0 +const lastUpdated = 1; +const defaultHotlist = { data: [] }; +const defaultStalelist = { + version: 2, + tolerance: 2, + lastUpdated, + eth_phishing_detect_config: { + fuzzylist: [], + allowlist: [], + blocklist: [], + name: ListNames.MetaMask, + }, + phishfort_hotlist: { + blocklist: [], + name: ListNames.Phishfort, + }, +}; + +const emptyHtmlPage = (blockProvider) => ` + + + + title + + + Empty page by ${blockProvider} + +`; + +/** + * Setup fetch mocks for the phishing detection feature. + * + * The mock configuration will show that "127.0.0.1" is blocked. The dynamic lookup on the warning + * page can be customized, so that we can test both the MetaMask and PhishFort block cases. + * + * @param {import('mockttp').Mockttp} mockServer - The mock server. + * @param {object} mockPhishingConfigResponseConfig - The response for the dynamic phishing + * @param {number} mockPhishingConfigResponseConfig.statusCode - The status code for the response. + * @param {string[]} mockPhishingConfigResponseConfig.blocklist - The blocklist for the response. + * @param {BlockProvider} mockPhishingConfigResponseConfig.blockProvider - The name of the provider who blocked the page. + * configuration lookup performed by the warning page. + */ +async function setupPhishingDetectionMocks( + mockServer, + { + statusCode = 200, + blocklist = ['127.0.0.1'], + blockProvider = BlockProvider.MetaMask, + }, +) { + const blockProviderConfig = resolveProviderConfigName(blockProvider); + + const response = + statusCode >= 400 + ? { statusCode } + : { + statusCode, + json: { + data: { + ...defaultStalelist, + [blockProviderConfig]: { + ...defaultStalelist[blockProviderConfig], + blocklist, + }, + }, + }, + }; + + await mockServer.forGet(METAMASK_STALELIST_URL).thenCallback(() => { + return response; + }); + + await mockServer + .forGet(`${METAMASK_HOTLIST_DIFF_URL}/${lastUpdated}`) + .thenCallback(() => { + return { + statusCode: 200, + json: defaultHotlist, + }; + }); + + await mockServer + .forGet('/~https://github.com/MetaMask/eth-phishing-detect/issues/new') + .thenCallback(() => { + return { + statusCode: 200, + body: emptyHtmlPage(blockProvider), + }; + }); + + await mockServer + .forGet('/~https://github.com/phishfort/phishfort-lists/issues/new') + .thenCallback(() => { + return { + statusCode: 200, + body: emptyHtmlPage(blockProvider), + }; + }); +} + +/** + * Mocks the request made from the phishing warning page to check eth-phishing-detect + * + * @param {*} mockServer + * @param {*} metamaskPhishingConfigResponse + */ +async function mockConfigLookupOnWarningPage( + mockServer, + metamaskPhishingConfigResponse, +) { + await mockServer + .forGet( + 'https://raw.githubusercontent.com/MetaMask/eth-phishing-detect/master/src/config.json', + ) + .thenCallback(() => metamaskPhishingConfigResponse); +} + +/** + * Setup fallback mocks for default behaviour of the phishing detection feature. + * + * This sets up default mocks for a mockttp server when included in test/e2e/mock-e2e.js + * + * @param {import('mockttp').Mockttp} mockServer - The mock server. + */ + +async function mockEmptyStalelistAndHotlist(mockServer) { + await mockServer.forGet(METAMASK_STALELIST_URL).thenCallback(() => { + return { + statusCode: 200, + json: { ...defaultStalelist }, + }; + }); + + await mockServer + .forGet(`${METAMASK_HOTLIST_DIFF_URL}/${lastUpdated}`) + .thenCallback(() => { + return { + statusCode: 200, + json: defaultHotlist, + }; + }); +} + +/** + * + * @param {BlockProvider} providerName - The name of the provider who issued the block. + * @returns {string} The name of the phishing config in the response. + */ +function resolveProviderConfigName(providerName) { + switch (providerName.toLowerCase()) { + case BlockProvider.MetaMask: + return 'eth_phishing_detect_config'; + case BlockProvider.PhishFort: + return 'phishfort_hotlist'; + default: + throw new Error('Provider name must either be metamask or phishfort'); + } +} + +module.exports = { + setupPhishingDetectionMocks, + mockEmptyStalelistAndHotlist, + mockConfigLookupOnWarningPage, +}; diff --git a/test/e2e/tests/phishing-detection.spec.js b/test/e2e/tests/phishing-controller/phishing-detection.spec.js similarity index 69% rename from test/e2e/tests/phishing-detection.spec.js rename to test/e2e/tests/phishing-controller/phishing-detection.spec.js index 735d2bfb2d92..2e29a2411ac3 100644 --- a/test/e2e/tests/phishing-detection.spec.js +++ b/test/e2e/tests/phishing-controller/phishing-detection.spec.js @@ -1,12 +1,17 @@ const { strict: assert } = require('assert'); + +const { convertToHexValue, withFixtures, openDapp } = require('../../helpers'); +const FixtureBuilder = require('../../fixture-builder'); +const { + METAMASK_HOTLIST_DIFF_URL, + METAMASK_STALELIST_URL, + BlockProvider, +} = require('./helpers'); + const { - convertToHexValue, - withFixtures, - openDapp, setupPhishingDetectionMocks, - mockPhishingDetection, -} = require('../helpers'); -const FixtureBuilder = require('../fixture-builder'); + mockConfigLookupOnWarningPage, +} = require('./mocks'); describe('Phishing Detection', function () { const ganacheOptions = { @@ -19,13 +24,32 @@ describe('Phishing Detection', function () { ], }; + describe('Phishing Detection Mock', function () { + it('should be updated to use v1 of the API', function () { + // Update the fixture in phishing-controller/mocks.js if this test fails + assert.equal( + METAMASK_STALELIST_URL, + 'https://phishing-detection.metafi.codefi.network/v1/stalelist', + ); + assert.equal( + METAMASK_HOTLIST_DIFF_URL, + 'https://phishing-detection.metafi.codefi.network/v1/diffsSince', + ); + }); + }); + it('should display the MetaMask Phishing Detection page and take the user to the blocked page if they continue', async function () { await withFixtures( { fixtures: new FixtureBuilder().build(), ganacheOptions, title: this.test.title, - testSpecificMock: mockPhishingDetection, + testSpecificMock: async (mockServer) => { + return setupPhishingDetectionMocks(mockServer, { + blockProvider: BlockProvider.MetaMask, + blocklist: ['127.0.0.1'], + }); + }, dapp: true, failOnConsoleError: false, }, @@ -44,12 +68,20 @@ describe('Phishing Detection', function () { }); it('should display the MetaMask Phishing Detection page in an iframe and take the user to the blocked page if they continue', async function () { + const DAPP_WITH_IFRAMED_PAGE_ON_BLOCKLIST = 'http://localhost:8080/'; + const IFRAMED_HOSTNAME = '127.0.0.1'; + await withFixtures( { fixtures: new FixtureBuilder().build(), ganacheOptions, title: this.test.title, - testSpecificMock: mockPhishingDetection, + testSpecificMock: async (mockServer) => { + return setupPhishingDetectionMocks(mockServer, { + blockProvider: BlockProvider.MetaMask, + blocklist: [IFRAMED_HOSTNAME], + }); + }, dapp: true, dappPaths: ['mock-page-with-iframe'], dappOptions: { @@ -61,7 +93,7 @@ describe('Phishing Detection', function () { await driver.navigate(); await driver.fill('#password', 'correct horse battery staple'); await driver.press('#password', driver.Key.ENTER); - await driver.openNewPage('http://localhost:8080/'); + await driver.openNewPage(DAPP_WITH_IFRAMED_PAGE_ON_BLOCKLIST); const iframe = await driver.findElement('iframe'); @@ -85,7 +117,12 @@ describe('Phishing Detection', function () { fixtures: new FixtureBuilder().build(), ganacheOptions, title: this.test.title, - testSpecificMock: mockPhishingDetection, + testSpecificMock: async (mockServer) => { + return setupPhishingDetectionMocks(mockServer, { + blockProvider: BlockProvider.MetaMask, + blocklist: ['127.0.0.1'], + }); + }, dapp: true, dappPaths: ['mock-page-with-disallowed-iframe'], dappOptions: { @@ -125,7 +162,11 @@ describe('Phishing Detection', function () { ganacheOptions, title: this.test.title, testSpecificMock: (mockServer) => { - setupPhishingDetectionMocks(mockServer, { statusCode: 500 }); + setupPhishingDetectionMocks(mockServer, { + blockProvider: BlockProvider.MetaMask, + blocklist: ['127.0.0.1'], + }); + mockConfigLookupOnWarningPage(mockServer, { statusCode: 500 }); }, dapp: true, failOnConsoleError: false, @@ -139,7 +180,9 @@ describe('Phishing Detection', function () { await driver.clickElement({ text: 'report a detection problem.' }); // wait for page to load before checking URL. - await driver.findElement({ text: 'Empty page' }); + await driver.findElement({ + text: `Empty page by ${BlockProvider.MetaMask}`, + }); assert.equal( await driver.getCurrentUrl(), `/~https://github.com/MetaMask/eth-phishing-detect/issues/new?title=[Legitimate%20Site%20Blocked]%20127.0.0.1&body=http%3A%2F%2F127.0.0.1%3A8080%2F`, @@ -149,12 +192,20 @@ describe('Phishing Detection', function () { }); it('should navigate the user to eth-phishing-detect to dispute a block from MetaMask', async function () { + // Must be site on actual eth-phishing-detect blocklist + const phishingSite = new URL('https://test.metamask-phishing.io'); + await withFixtures( { fixtures: new FixtureBuilder().build(), ganacheOptions, title: this.test.title, - testSpecificMock: mockPhishingDetection, + testSpecificMock: async (mockServer) => { + return setupPhishingDetectionMocks(mockServer, { + blockProvider: BlockProvider.MetaMask, + blocklist: [phishingSite.hostname], + }); + }, dapp: true, failOnConsoleError: false, }, @@ -162,37 +213,34 @@ describe('Phishing Detection', function () { await driver.navigate(); await driver.fill('#password', 'correct horse battery staple'); await driver.press('#password', driver.Key.ENTER); - await openDapp(driver); + await driver.openNewPage(phishingSite.href); await driver.clickElement({ text: 'report a detection problem.' }); // wait for page to load before checking URL. - await driver.findElement({ text: 'Empty page' }); + await driver.findElement({ + text: `Empty page by ${BlockProvider.MetaMask}`, + }); assert.equal( await driver.getCurrentUrl(), - `/~https://github.com/MetaMask/eth-phishing-detect/issues/new?title=[Legitimate%20Site%20Blocked]%20127.0.0.1&body=http%3A%2F%2F127.0.0.1%3A8080%2F`, + `/~https://github.com/MetaMask/eth-phishing-detect/issues/new?title=[Legitimate%20Site%20Blocked]%20${encodeURIComponent( + phishingSite.hostname, + )}&body=${encodeURIComponent(phishingSite.href)}`, ); }, ); }); - it('should navigate the user to PhishFort to dispute a block from MetaMask', async function () { + it('should navigate the user to PhishFort to dispute a Phishfort Block', async function () { await withFixtures( { fixtures: new FixtureBuilder().build(), ganacheOptions, title: this.test.title, - testSpecificMock: (mockServer) => { - setupPhishingDetectionMocks(mockServer, { - statusCode: 200, - json: { - version: 2, - tolerance: 2, - fuzzylist: [], - whitelist: [], - blacklist: [], - lastUpdated: 0, - }, + testSpecificMock: async (mockServer) => { + return setupPhishingDetectionMocks(mockServer, { + blockProvider: BlockProvider.PhishFort, + blocklist: ['127.0.0.1'], }); }, dapp: true, @@ -202,12 +250,14 @@ describe('Phishing Detection', function () { await driver.navigate(); await driver.fill('#password', 'correct horse battery staple'); await driver.press('#password', driver.Key.ENTER); - await openDapp(driver); + await driver.openNewPage('http://127.0.0.1:8080'); await driver.clickElement({ text: 'report a detection problem.' }); // wait for page to load before checking URL. - await driver.findElement({ text: 'Empty page' }); + await driver.findElement({ + text: `Empty page by ${BlockProvider.PhishFort}`, + }); assert.equal( await driver.getCurrentUrl(), `/~https://github.com/phishfort/phishfort-lists/issues/new?title=[Legitimate%20Site%20Blocked]%20127.0.0.1&body=http%3A%2F%2F127.0.0.1%3A8080%2F`, @@ -222,7 +272,12 @@ describe('Phishing Detection', function () { fixtures: new FixtureBuilder().build(), ganacheOptions, title: this.test.title, - testSpecificMock: mockPhishingDetection, + testSpecificMock: async (mockServer) => { + return setupPhishingDetectionMocks(mockServer, { + blockProvider: BlockProvider.MetaMask, + blocklist: ['127.0.0.1'], + }); + }, dapp: true, dappPaths: ['mock-page-with-disallowed-iframe'], dappOptions: { diff --git a/yarn.lock b/yarn.lock index 5bd48d075bed..2d7aa26eb1e2 100644 --- a/yarn.lock +++ b/yarn.lock @@ -3909,7 +3909,7 @@ __metadata: languageName: node linkType: hard -"@metamask/controller-utils@npm:^3.0.0, @metamask/controller-utils@npm:^3.4.0": +"@metamask/controller-utils@npm:^3.0.0, @metamask/controller-utils@npm:^3.1.0, @metamask/controller-utils@npm:^3.4.0": version: 3.4.0 resolution: "@metamask/controller-utils@npm:3.4.0" dependencies: @@ -4494,16 +4494,16 @@ __metadata: languageName: node linkType: hard -"@metamask/phishing-controller@npm:^3.0.0": - version: 3.0.0 - resolution: "@metamask/phishing-controller@npm:3.0.0" +"@metamask/phishing-controller@npm:^4.0.0": + version: 4.0.0 + resolution: "@metamask/phishing-controller@npm:4.0.0" dependencies: "@metamask/base-controller": ^2.0.0 - "@metamask/controller-utils": ^3.0.0 + "@metamask/controller-utils": ^3.1.0 "@types/punycode": ^2.1.0 eth-phishing-detect: ^1.2.0 punycode: ^2.1.1 - checksum: b0b9a86cba1928f0fd22a2aed196d75dc19a5e56547efe1b533d7ae06eaaf9432a6ee5004a8fd477f52310b50c2f3635a1e70ac83e3670f4cc6a1f488a674d73 + checksum: 15de581f7bec21d75531167275c68d7bbeae7fdaad02268749ba0a71c4d3ccb53718d963d6583e90c337407f65b7fcc9a89eb76c6f731802c2668a8425d5df89 languageName: node linkType: hard @@ -24273,7 +24273,7 @@ __metadata: "@metamask/notification-controller": ^3.0.0 "@metamask/obs-store": ^8.1.0 "@metamask/permission-controller": ^4.0.0 - "@metamask/phishing-controller": ^3.0.0 + "@metamask/phishing-controller": ^4.0.0 "@metamask/phishing-warning": ^2.1.0 "@metamask/post-message-stream": ^6.0.0 "@metamask/ppom-validator": ^0.1.2 From d3236b3b42b9a6e5ce14bc5bb3549d3575ca7863 Mon Sep 17 00:00:00 2001 From: Peter <53189696+PeterYinusa@users.noreply.github.com> Date: Mon, 31 Jul 2023 10:32:30 -0400 Subject: [PATCH 16/25] [skip e2e] document ens flows (#20261) --- test/scenarios/5. ens/name resolution.csv | 20 ++++++++++++++++++++ 1 file changed, 20 insertions(+) create mode 100644 test/scenarios/5. ens/name resolution.csv diff --git a/test/scenarios/5. ens/name resolution.csv b/test/scenarios/5. ens/name resolution.csv new file mode 100644 index 000000000000..2f9aa182568a --- /dev/null +++ b/test/scenarios/5. ens/name resolution.csv @@ -0,0 +1,20 @@ +Step,Test Steps,Test Data,Expected Result,Notes +1,Open the extension.,,The Welcome Back screen is shown., +2,Proceed to Unlock the wallet.,password (8 characters min).,"The Ether balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Ethereum Mainnet.", +3,Switch networks to a test network.,e.g. Goerli.,"The Goerli balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Goerli.", +4,Proceed to the Send flow.,,, +5,Enter an ENS address.,e.g. peteryinusa.eth.,"A matching result appears in the recipient list. +The ENS address is shown in the list item. +The hexadecimal address is shown in the list item.", +6,Select the matching recipient.,,"The send screen is shown. +The ENS address is shown in the recipient field. +The hexadecimal address is shown in the recipient field.", +7,Enter an amount.,e.g. 0.1.,, +8,Proceed to the confirmation screen.,,The recipient's ENS address is shown., +9,Confirm the transaction.,,"The transaction appears in the activity list. +The recipient's hexadecimal address is shown in the activity list item.", +10,Open the activity list item.,,The recipient's ENS address is shown in the item details., \ No newline at end of file From 5df6a71ceae232b5b30b2dae488ac1fcb8f4709f Mon Sep 17 00:00:00 2001 From: cryptodev-2s <109512101+cryptodev-2s@users.noreply.github.com> Date: Mon, 31 Jul 2023 16:45:20 +0100 Subject: [PATCH 17/25] Use getKeyringsByType from core KeyringController (#20210) --- app/scripts/metamask-controller.js | 21 ++++++++++++--------- app/scripts/metamask-controller.test.js | 14 +++++++------- 2 files changed, 19 insertions(+), 16 deletions(-) diff --git a/app/scripts/metamask-controller.js b/app/scripts/metamask-controller.js index 9efbb8447fd2..7cf31bdb5f68 100644 --- a/app/scripts/metamask-controller.js +++ b/app/scripts/metamask-controller.js @@ -1817,7 +1817,7 @@ export default class MetamaskController extends EventEmitter { */ async getSnapKeyring() { if (!this.snapKeyring) { - let [snapKeyring] = this.keyringController.getKeyringsByType( + let [snapKeyring] = this.coreKeyringController.getKeyringsByType( KeyringType.snap, ); if (!snapKeyring) { @@ -2932,7 +2932,7 @@ export default class MetamaskController extends EventEmitter { ethQuery, ); - const [primaryKeyring] = keyringController.getKeyringsByType( + const [primaryKeyring] = this.coreKeyringController.getKeyringsByType( KeyringType.hdKeyTree, ); if (!primaryKeyring) { @@ -3122,7 +3122,7 @@ export default class MetamaskController extends EventEmitter { * Gets the mnemonic of the user's primary keyring. */ getPrimaryKeyringMnemonic() { - const [keyring] = this.keyringController.getKeyringsByType( + const [keyring] = this.coreKeyringController.getKeyringsByType( KeyringType.hdKeyTree, ); if (!keyring.mnemonic) { @@ -3137,7 +3137,8 @@ export default class MetamaskController extends EventEmitter { const custodyType = this.custodyController.getCustodyTypeByAddress( toChecksumHexAddress(address), ); - const keyring = this.keyringController.getKeyringsByType(custodyType)[0]; + const keyring = + this.coreKeyringController.getKeyringsByType(custodyType)[0]; return keyring?.getAccountDetails(address) ? keyring : undefined; } ///: END:ONLY_INCLUDE_IN @@ -3174,7 +3175,9 @@ export default class MetamaskController extends EventEmitter { 'MetamaskController:getKeyringForDevice - Unknown device', ); } - let [keyring] = await this.keyringController.getKeyringsByType(keyringName); + let [keyring] = await this.coreKeyringController.getKeyringsByType( + keyringName, + ); if (!keyring) { keyring = await this.keyringController.addNewKeyring(keyringName); } @@ -3387,7 +3390,7 @@ export default class MetamaskController extends EventEmitter { await new Promise((resolve) => setTimeout(resolve, 5_000)); } - const [primaryKeyring] = this.keyringController.getKeyringsByType( + const [primaryKeyring] = this.coreKeyringController.getKeyringsByType( KeyringType.hdKeyTree, ); if (!primaryKeyring) { @@ -3432,7 +3435,7 @@ export default class MetamaskController extends EventEmitter { * encoded as an array of UTF-8 bytes. */ async verifySeedPhrase() { - const [primaryKeyring] = this.keyringController.getKeyringsByType( + const [primaryKeyring] = this.coreKeyringController.getKeyringsByType( KeyringType.hdKeyTree, ); if (!primaryKeyring) { @@ -4649,14 +4652,14 @@ export default class MetamaskController extends EventEmitter { * Locks MetaMask */ setLocked() { - const [trezorKeyring] = this.keyringController.getKeyringsByType( + const [trezorKeyring] = this.coreKeyringController.getKeyringsByType( KeyringType.trezor, ); if (trezorKeyring) { trezorKeyring.dispose(); } - const [ledgerKeyring] = this.keyringController.getKeyringsByType( + const [ledgerKeyring] = this.coreKeyringController.getKeyringsByType( KeyringType.ledger, ); ledgerKeyring?.destroy?.(); diff --git a/app/scripts/metamask-controller.test.js b/app/scripts/metamask-controller.test.js index 5bed8c899d51..b29fce443d78 100644 --- a/app/scripts/metamask-controller.test.js +++ b/app/scripts/metamask-controller.test.js @@ -310,9 +310,9 @@ describe('MetaMaskController', function () { ]); }); - it('adds private key to keyrings in KeyringController', async function () { + it('adds private key to keyrings in core KeyringController', async function () { const simpleKeyrings = - metamaskController.keyringController.getKeyringsByType( + metamaskController.coreKeyringController.getKeyringsByType( KeyringType.imported, ); const pubAddressHexArr = await simpleKeyrings[0].getAccounts(); @@ -595,7 +595,7 @@ describe('MetaMaskController', function () { .connectHardware(HardwareDeviceNames.trezor, 0) .catch(() => null); const keyrings = - await metamaskController.keyringController.getKeyringsByType( + await metamaskController.coreKeyringController.getKeyringsByType( KeyringType.trezor, ); assert.deepEqual( @@ -611,7 +611,7 @@ describe('MetaMaskController', function () { .connectHardware(HardwareDeviceNames.ledger, 0) .catch(() => null); const keyrings = - await metamaskController.keyringController.getKeyringsByType( + await metamaskController.coreKeyringController.getKeyringsByType( KeyringType.ledger, ); assert.deepEqual( @@ -638,7 +638,7 @@ describe('MetaMaskController', function () { mnemonic: uint8ArrayMnemonic, }; sinon - .stub(metamaskController.keyringController, 'getKeyringsByType') + .stub(metamaskController.coreKeyringController, 'getKeyringsByType') .returns([mockHDKeyring]); const recoveredMnemonic = @@ -692,7 +692,7 @@ describe('MetaMaskController', function () { .catch(() => null); await metamaskController.forgetDevice(HardwareDeviceNames.trezor); const keyrings = - await metamaskController.keyringController.getKeyringsByType( + await metamaskController.coreKeyringController.getKeyringsByType( KeyringType.trezor, ); @@ -755,7 +755,7 @@ describe('MetaMaskController', function () { it('should set unlockedAccount in the keyring', async function () { const keyrings = - await metamaskController.keyringController.getKeyringsByType( + await metamaskController.coreKeyringController.getKeyringsByType( KeyringType.trezor, ); assert.equal(keyrings[0].unlockedAccount, accountToUnlock); From e2e1a5fe0773e49df3da3be2e2bb6900aace4909 Mon Sep 17 00:00:00 2001 From: Brad Decker Date: Mon, 31 Jul 2023 14:51:39 -0500 Subject: [PATCH 18/25] Use node 18 (#20308) * use node v18 --- .circleci/config.yml | 6 +++--- .nvmrc | 2 +- README.md | 6 +++--- package.json | 2 +- 4 files changed, 8 insertions(+), 8 deletions(-) diff --git a/.circleci/config.yml b/.circleci/config.yml index 8b05dfcaab34..ffc009b2073f 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -3,16 +3,16 @@ version: 2.1 executors: node-browsers: docker: - - image: cimg/node:16.20-browsers + - image: cimg/node:18.17-browsers node-browsers-medium-plus: docker: - - image: cimg/node:16.20-browsers + - image: cimg/node:18.17-browsers resource_class: medium+ environment: NODE_OPTIONS: --max_old_space_size=2048 node-browsers-large: docker: - - image: cimg/node:16.20-browsers + - image: cimg/node:18.17-browsers resource_class: large environment: NODE_OPTIONS: --max_old_space_size=2048 diff --git a/.nvmrc b/.nvmrc index 6f7f377bf514..3f430af82b3d 100644 --- a/.nvmrc +++ b/.nvmrc @@ -1 +1 @@ -v16 +v18 diff --git a/README.md b/README.md index 38d94d726f74..b8e257d1dfb4 100644 --- a/README.md +++ b/README.md @@ -14,11 +14,11 @@ To learn how to contribute to the MetaMask project itself, visit our [Internal D ## Building locally -- Install [Node.js](https://nodejs.org) version 16 +- Install [Node.js](https://nodejs.org) version 18 - If you are using [nvm](/~https://github.com/nvm-sh/nvm#installing-and-updating) (recommended) running `nvm use` will automatically choose the right node version for you. - Install [Yarn v3](https://yarnpkg.com/getting-started/install) - - ONLY follow the steps in the "Install Corepack" and "Updating the global Yarn version" sections - - DO NOT take any of the steps in the "Initializing your project", "Updating to the latest versions" or "Installing the latest build fresh from master" sections. These steps could result in your repo being reset or installing the wrong yarn version, which can break your build. + - ONLY follow the steps in the "Install Corepack" and "Updating the global Yarn version" sections + - DO NOT take any of the steps in the "Initializing your project", "Updating to the latest versions" or "Installing the latest build fresh from master" sections. These steps could result in your repo being reset or installing the wrong yarn version, which can break your build. - Duplicate `.metamaskrc.dist` within the root and rename it to `.metamaskrc` - Replace the `INFURA_PROJECT_ID` value with your own personal [Infura Project ID](https://infura.io/docs). - If debugging MetaMetrics, you'll need to add a value for `SEGMENT_WRITE_KEY` [Segment write key](https://segment.com/docs/connections/find-writekey/), see [Developing on MetaMask - Segment](./development/README.md#segment). diff --git a/package.json b/package.json index 0261e26af05e..0cc86e215d9c 100644 --- a/package.json +++ b/package.json @@ -559,7 +559,7 @@ "yargs": "^17.0.1" }, "engines": { - "node": "^16.20.0", + "node": ">= 18", "yarn": "^3.2.4" }, "lavamoat": { From 2d412c21bbcf79257aefc98f2e86915bc6dc0459 Mon Sep 17 00:00:00 2001 From: Peter <53189696+PeterYinusa@users.noreply.github.com> Date: Mon, 31 Jul 2023 15:55:33 -0400 Subject: [PATCH 19/25] [skip e2e] document phishing flows (#20264) --- test/scenarios/6. phishing/warning page.csv | 12 ++++++++++++ 1 file changed, 12 insertions(+) create mode 100644 test/scenarios/6. phishing/warning page.csv diff --git a/test/scenarios/6. phishing/warning page.csv b/test/scenarios/6. phishing/warning page.csv new file mode 100644 index 000000000000..9491ef5181c9 --- /dev/null +++ b/test/scenarios/6. phishing/warning page.csv @@ -0,0 +1,12 @@ +Step,Test Steps,Test Data,Expected Result,Notes +1,Open the extension.,,The Welcome Back screen is shown., +2,Proceed to Unlock the wallet.,password (8 characters min).,"The Ether balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Ethereum Mainnet.", +3,Navigate to a block list site.,"See the block list section of config.json in +/~https://github.com/MetaMask/eth-phishing-detect.",The warning page is displayed., +4,Proceed Back to safety.,,Redirected to the expanded extension overview page., +5,Navigate to a block list site.,"See the block list section of config.json in +/~https://github.com/MetaMask/eth-phishing-detect.",The warning page is displayed., +6,Proceed to the block list site.,,Redirected to the block list site.,Do not interact with the block list site. +7,Close the block list site.,,, \ No newline at end of file From 2daf00224533e52f4797fbc36ad1130963e6ba18 Mon Sep 17 00:00:00 2001 From: Peter <53189696+PeterYinusa@users.noreply.github.com> Date: Mon, 31 Jul 2023 15:56:14 -0400 Subject: [PATCH 20/25] [skip e2e] document token flows (#20260) --- .../import erc1155 token origin MM.csv | 35 +++++++++++++++++ .../import erc20 token origin MM.csv | 38 +++++++++++++++++++ .../import erc721 token origin MM.csv | 35 +++++++++++++++++ 3 files changed, 108 insertions(+) create mode 100644 test/scenarios/4. tokens/import erc1155 token origin MM.csv create mode 100644 test/scenarios/4. tokens/import erc20 token origin MM.csv create mode 100644 test/scenarios/4. tokens/import erc721 token origin MM.csv diff --git a/test/scenarios/4. tokens/import erc1155 token origin MM.csv b/test/scenarios/4. tokens/import erc1155 token origin MM.csv new file mode 100644 index 000000000000..1c4218d7cf42 --- /dev/null +++ b/test/scenarios/4. tokens/import erc1155 token origin MM.csv @@ -0,0 +1,35 @@ +Step,Test Steps,Test Data,Expected Result,Notes +1,Open the extension.,,The Welcome Back screen is shown., +2,Proceed to Unlock the wallet.,password (8 characters min).,"The Ether balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Ethereum Mainnet.", +3,Switch networks to a test network.,e.g. Goerli.,"The Goerli balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Goerli.", +4,Open the test dapp in another tab.,https://metamask.github.io/test-dapp/,, +5,Proceed to Connect with MetaMask.,,, +6,Connect with the current account.,,, +7,Deploy the ERC1155 contract.,,, +8,Confirm the transaction and wait for the it to complete.,,, +9,Proceed to Batch Mint NFTs.,,, +10,Confirm the transaction and wait for the it to complete.,,, +11,Close the test dapp and switch back to the wallet.,,, +12,Open the activity list item.,,"The transaction status, recipient's address, nonce, amount, gas and total are shown in the item details.", +13,Proceed to view the transaction on the block explorer.,,The block explorer opens in a new tab., Take a note of the Contract address. +14,Close the block explorer and switch back to the wallet.,,, +15,Switch to the NFTs tab.,,, +16,Proceed to the Import tokens flow.,,, +17,Enter the contract address.,Contract address.,, +18,Enter the token id.,e.g. 1.,, +19,Confirm the addition of the custom token.,,"The NFT is shown in the wallets NFTs list. +The NFT image is shown.", +20,Proceed to open the NFT.,,"The NFT details page is shown. +The NFT collection name is shown. +The NFT token id is shown. +The NFT description is shown. +The NFT image source is shown. +The NFT contract address is shown.", +21,Open the asset menu.,,, +22,Proceed to view the asset on Opensea.,,Opensea opens in a new tab., +23,Close Opensea and switch back to the wallet.,,, +24,Navigate to the overview page using the asset breadcrumbs.,,The overview page is shown., \ No newline at end of file diff --git a/test/scenarios/4. tokens/import erc20 token origin MM.csv b/test/scenarios/4. tokens/import erc20 token origin MM.csv new file mode 100644 index 000000000000..4a7c2c2aca62 --- /dev/null +++ b/test/scenarios/4. tokens/import erc20 token origin MM.csv @@ -0,0 +1,38 @@ +Step,Test Steps,Test Data,Expected Result,Notes +1,Open the extension.,,The Welcome Back screen is shown., +2,Proceed to Unlock the wallet.,password (8 characters min).,"The Ether balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Ethereum Mainnet.", +3,Switch networks to a test network.,e.g. Goerli.,"The Goerli balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Goerli.", +4,Open the test dapp in another tab.,https://metamask.github.io/test-dapp/,, +5,Proceed to Connect with MetaMask.,,, +6,Connect with the current account.,,, +7,Deploy the Token contract.,,, +8,Confirm the transaction and wait for the it to complete.,,, +9,Close the test dapp and switch back to the wallet.,,, +10,Open the activity list item.,,"The transaction status, recipient's address, nonce, amount, gas and total are shown in the item details.", +11,Proceed to view the transaction on the block explorer.,,The block explorer opens in a new tab., Take a note of the Contract address. +12,Close the block explorer and switch back to the wallet.,,, +13,Switch to the Tokens tab.,,, +14,Proceed to the Import tokens flow.,,, +15,Enter the contract address.,Contract address.,, +16,Enter the token symbol.,e.g. TST.,, +17,Enter the token decimals.,e.g. 4.,, +18,Confirm the addition of the custom token.,,"The asset page is shown. +The specified token symbol is shown. +The specified token icon is shown. +The specified token balance is shown.", +19,Open the asset menu.,,, +20,Proceed to view the asset in the block explorer.,,The block explorer opens in a new tab., +21,Close the block explorer and switch back to the wallet.,,, +22,Open the asset menu.,,, +23,Proceed to view the token details.,,"The token details page is shown. +The token balance is shown. +The token contract symbol is shown. +The token decimals is shown. +The current network is shown.", +24,Close the token details page.,,, +25,Navigate to the overview page using the asset breadcrumbs.,,"The overview page is shown. +The TST token is shown in the wallets token list.", \ No newline at end of file diff --git a/test/scenarios/4. tokens/import erc721 token origin MM.csv b/test/scenarios/4. tokens/import erc721 token origin MM.csv new file mode 100644 index 000000000000..f4bab1bae136 --- /dev/null +++ b/test/scenarios/4. tokens/import erc721 token origin MM.csv @@ -0,0 +1,35 @@ +Step,Test Steps,Test Data,Expected Result,Notes +1,Open the extension.,,The Welcome Back screen is shown., +2,Proceed to Unlock the wallet.,password (8 characters min).,"The Ether balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Ethereum Mainnet.", +3,Switch networks to a test network.,e.g. Goerli.,"The Goerli balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Goerli.", +4,Open the test dapp in another tab.,https://metamask.github.io/test-dapp/,, +5,Proceed to Connect with MetaMask.,,, +6,Connect with the current account.,,, +7,Deploy the NFT contract.,,, +8,Confirm the transaction and wait for the it to complete.,,, +9,Proceed to Mint an NFT.,,, +10,Confirm the transaction and wait for the it to complete.,,, +11,Close the test dapp and switch back to the wallet.,,, +12,Open the activity list item.,,"The transaction status, recipient's address, nonce, amount, gas and total are shown in the item details.", +13,Proceed to view the transaction on the block explorer.,,The block explorer opens in a new tab., Take a note of the Contract address. +14,Close the block explorer and switch back to the wallet.,,, +15,Switch to the NFTs tab.,,, +16,Proceed to the Import tokens flow.,,, +17,Enter the contract address.,Contract address.,, +18,Enter the token id.,e.g. 1.,, +19,Confirm the addition of the custom token.,,"The NFT is shown in the wallets NFTs list. +The NFT image is shown.", +20,Proceed to open the NFT.,,"The NFT details page is shown. +The NFT collection name is shown. +The NFT token id is shown. +The NFT description is shown. +The NFT image source is shown. +The NFT contract address is shown.", +21,Open the asset menu.,,, +22,Proceed to view the asset on Opensea.,,Opensea opens in a new tab., +23,Close Opensea and switch back to the wallet.,,, +24,Navigate to the overview page using the asset breadcrumbs.,,The overview page is shown., \ No newline at end of file From 523b3145b26764cbda1174e8676b0db9cd03eeae Mon Sep 17 00:00:00 2001 From: Peter <53189696+PeterYinusa@users.noreply.github.com> Date: Mon, 31 Jul 2023 16:10:50 -0400 Subject: [PATCH 21/25] [skip e2e] document transaction flows (#20259) --- .../3. transactions/cancel transaction.csv | 30 +++++++++++++++ .../send erc20 token origin MM.csv | 37 +++++++++++++++++++ .../send erc721 token origin MM.csv | 37 +++++++++++++++++++ .../send native token origin MM.csv | 21 +++++++++++ .../3. transactions/speed up transaction.csv | 27 ++++++++++++++ 5 files changed, 152 insertions(+) create mode 100644 test/scenarios/3. transactions/cancel transaction.csv create mode 100644 test/scenarios/3. transactions/send erc20 token origin MM.csv create mode 100644 test/scenarios/3. transactions/send erc721 token origin MM.csv create mode 100644 test/scenarios/3. transactions/send native token origin MM.csv create mode 100644 test/scenarios/3. transactions/speed up transaction.csv diff --git a/test/scenarios/3. transactions/cancel transaction.csv b/test/scenarios/3. transactions/cancel transaction.csv new file mode 100644 index 000000000000..852cd56f947a --- /dev/null +++ b/test/scenarios/3. transactions/cancel transaction.csv @@ -0,0 +1,30 @@ +Step,Test Steps,Test Data,Expected Result,Notes +1,Open the extension.,,The Welcome Back screen is shown., +2,Proceed to Unlock the wallet.,password (8 characters min).,"The Ether balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Ethereum Mainnet.", +3,Switch networks to a test network.,e.g. Goerli.,"The Goerli balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Goerli.", +4,Proceed to the Send flow.,,, +5,Enter a public address.,e.g. 0x56F2e03c8D30649818c022a9759CF43B240D08B3.,, +6,Enter an amount.,e.g. 0.1.,, +7,Proceed to the confirmation screen.,,"The recipient's hexadecimal address is shown. +The specified amount is shown. +The estimated gas details are shown. +The total amount is shown.", +8,Proceed to edit the transaction gas fees.,e.g. 0.1.,, +9,Reduce the the Max base fee and Priority fee.,e.g. Max base fee 0.00001 and Priority fee 0.00001.,, +10,Confirm the transaction.,,"The transaction appears in the activity list. +The recipient's hexadecimal address is shown in the activity list item. +The amount is shown in the activity list item.", +11,Cancel the transaction.,,"The transaction appears in the activity list. +The recipient's hexadecimal address is shown in the activity list item. +The amount is shown in the activity list item.", +12,Wait for the transaction to complete.,,Another activity item appears.,Two activity items; Send and Receive. +13,Open the Send activity list item.,,"The updated gas fees are shown in the item details. +The transaction status, recipient's address, nonce, amount, gas and total are shown in the item details.", +14,Expand the Activity log.,,"The created, submitted, cancel attempt and cancelled activity is shown in the activity log.", +15,Proceed to view the transaction on the block explorer.,,The block explorer opens in a new tab., +16,Open the Receive activity list item.,,"The transaction status, recipient's address, nonce, amount, gas and total are shown in the item details.", +17,Proceed to view the transaction on the block explorer.,,The block explorer opens in a new tab., \ No newline at end of file diff --git a/test/scenarios/3. transactions/send erc20 token origin MM.csv b/test/scenarios/3. transactions/send erc20 token origin MM.csv new file mode 100644 index 000000000000..b7d6ad181222 --- /dev/null +++ b/test/scenarios/3. transactions/send erc20 token origin MM.csv @@ -0,0 +1,37 @@ +Step,Test Steps,Test Data,Expected Result,Notes +1,Open the extension.,,The Welcome Back screen is shown., +2,Proceed to Unlock the wallet.,password (8 characters min).,"The Ether balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Ethereum Mainnet.", +3,Switch networks to a test network.,e.g. Goerli.,"The Goerli balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Goerli.", +4,Open the test dapp in another tab.,https://metamask.github.io/test-dapp/,, +5,Proceed to Connect with MetaMask.,,, +6,Connect with the current account.,,, +7,Deploy the Token contract.,,, +8,Confirm the transaction and wait for the it to complete.,,, +9,Proceed to add the token to the wallet.,,, +10,Confirm the suggested token notification.,,"The TST token is shown in the wallets token list. +The TST token symbol is shown. +The TST token icon is shown. +The TST token balance is shown.", +11,Close the test dapp and switch back to the wallet.,,, +12,Proceed to the Send flow.,,, +13,Enter a public address.,e.g. 0x56F2e03c8D30649818c022a9759CF43B240D08B3.,, +14,Change the asset to the newly added token.,e.g. TST.,"The TST token is shown in the asset field on the send flow. +The TST token symbol is shown. +The TST token icon is shown. +The TST token balance is shown.", +15,Enter an amount.,e.g. 0.1.,, +16,Proceed to the confirmation screen.,,"The recipient's hexadecimal address is shown. +The specified amount is shown. +The specified token is shown. +The estimated gas details are shown. +The total amount is shown.", +17,Confirm the transaction.,,"The transaction appears in the activity list. +The recipient's hexadecimal address is shown in the activity list item. +The amount is shown in the activity list item.", +18,Open the activity list item.,,"The transaction status, recipient's address, nonce, amount, gas and total are shown in the item details.", +19,Expand the Activity log.,,"The created, submitted and confirmed activity is shown in the activity log.", +20,Proceed to view the transaction on the block explorer.,,The block explorer opens in a new tab., \ No newline at end of file diff --git a/test/scenarios/3. transactions/send erc721 token origin MM.csv b/test/scenarios/3. transactions/send erc721 token origin MM.csv new file mode 100644 index 000000000000..af4bc2d6739c --- /dev/null +++ b/test/scenarios/3. transactions/send erc721 token origin MM.csv @@ -0,0 +1,37 @@ +Step,Test Steps,Test Data,Expected Result,Notes +1,Open the extension.,,The Welcome Back screen is shown., +2,Proceed to Unlock the wallet.,password (8 characters min).,"The Ether balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Ethereum Mainnet.", +3,Switch networks to a test network.,e.g. Goerli.,"The Goerli balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Goerli.", +4,Open the test dapp in another tab.,https://metamask.github.io/test-dapp/,, +5,Proceed to Connect with MetaMask.,,, +6,Connect with the current account.,,, +7,Proceed to Deploy the NFT contract.,,, +8,Confirm the transaction and wait for the it to complete.,,, +9,Proceed to Mint an NFT.,,, +10,Confirm the transaction and wait for the it to complete.,,, +11,Proceed to Watch NFT.,,, +12,Confirm the suggested NFT notification.,,"The NFT is shown in the wallets NFTs list. +The NFT image is shown.", +13,Close the test dapp and switch back to the wallet.,,, +14,Proceed to the Send flow.,,, +15,Enter a public address.,e.g. 0x56F2e03c8D30649818c022a9759CF43B240D08B3.,, +16,Change the asset to the newly added NFT.,e.g. TestDappNFTs.,"The TestDappNFTs NFT is shown in the asset field on the send flow. +The specified NFT symbol is shown. +The specified NFT image is shown. +The specified NFT token id is shown.", +17,Proceed to the confirmation screen.,,"The recipient's hexadecimal address is shown. +The specified NFT symbol is shown. +The specified NFT image is shown. +The specified NFT token id is shown. +The estimated gas details are shown. +The total amount is shown.", +18,Confirm the transaction.,,"The transaction appears in the activity list. +The recipient's hexadecimal address is shown in the activity list item. +The amount is shown in the activity list item.", +19,Open the activity list item.,,"The transaction status, recipient's address, nonce, amount, gas and total are shown in the item details.", +20,Expand the Activity log.,,"The created, submitted and confirmed activity is shown in the activity log.", +21,Proceed to view the transaction on the block explorer.,,The block explorer opens in a new tab., \ No newline at end of file diff --git a/test/scenarios/3. transactions/send native token origin MM.csv b/test/scenarios/3. transactions/send native token origin MM.csv new file mode 100644 index 000000000000..8bf34a45033a --- /dev/null +++ b/test/scenarios/3. transactions/send native token origin MM.csv @@ -0,0 +1,21 @@ +Step,Test Steps,Test Data,Expected Result,Notes +1,Open the extension.,,The Welcome Back screen is shown., +2,Proceed to Unlock the wallet.,password (8 characters min).,"The Ether balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Ethereum Mainnet.", +3,Switch networks to a test network.,e.g. Goerli.,"The Goerli balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Goerli.", +4,Proceed to the Send flow.,,, +5,Enter a public address.,e.g. 0x56F2e03c8D30649818c022a9759CF43B240D08B3.,, +6,Enter an amount.,e.g. 0.1.,, +7,Proceed to the confirmation screen.,,"The recipient's hexadecimal address is shown. +The specified amount is shown. +The estimated gas details are shown. +The total amount is shown.", +8,Confirm the transaction.,,"The transaction appears in the activity list. +The recipient's hexadecimal address is shown in the activity list item. +The amount is shown in the activity list item.", +9,Open the activity list item.,,"The transaction status, recipient's address, nonce, amount, gas and total are shown in the item details.", +10,Expand the Activity log.,,"The created, submitted and confirmed activity is shown in the activity log.", +11,Proceed to view the transaction on the block explorer.,,The block explorer opens in a new tab., \ No newline at end of file diff --git a/test/scenarios/3. transactions/speed up transaction.csv b/test/scenarios/3. transactions/speed up transaction.csv new file mode 100644 index 000000000000..5e89b3a52641 --- /dev/null +++ b/test/scenarios/3. transactions/speed up transaction.csv @@ -0,0 +1,27 @@ +Step,Test Steps,Test Data,Expected Result,Notes +1,Open the extension.,,The Welcome Back screen is shown., +2,Proceed to Unlock the wallet.,password (8 characters min).,"The Ether balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Ethereum Mainnet.", +3,Switch networks to a test network.,e.g. Goerli.,"The Goerli balance is shown on the overview. +The wallet address is shown on the overview. +The selected network is Goerli.", +4,Proceed to the Send flow.,,, +5,Enter a public address.,e.g. 0x56F2e03c8D30649818c022a9759CF43B240D08B3.,, +6,Enter an amount.,e.g. 0.1.,, +7,Proceed to the confirmation screen.,,"The recipient's hexadecimal address is shown. +The specified amount is shown. +The estimated gas details are shown. +The total amount is shown.", +8,Proceed to edit the transaction gas fees.,e.g. 0.1.,, +9,Reduce the the Max base fee and Priority fee.,e.g. Max base fee 0.00001 and Priority fee 0.00001.,, +10,Confirm the transaction.,,"The transaction appears in the activity list. +The recipient's hexadecimal address is shown in the activity list item. +The amount is shown in the activity list item.", +11,Speed up the transaction.,,"The transaction appears in the activity list. +The recipient's hexadecimal address is shown in the activity list item. +The amount is shown in the activity list item.", +12,Open the activity list item.,,"The updated gas fees are shown in the item details. +The transaction status, recipient's address, nonce, amount, gas and total are shown in the item details.", +13,Expand the Activity log.,,"The created, submitted, resubmitted and confirmed activity is shown in the activity log.", +14,Proceed to view the transaction on the block explorer.,,The block explorer opens in a new tab., \ No newline at end of file From 507c2cb475a8a71b848a917d82975b277f9f9603 Mon Sep 17 00:00:00 2001 From: Mark Stacey Date: Mon, 31 Jul 2023 18:49:32 -0230 Subject: [PATCH 22/25] Capture Sentry errors prior to initialization (#20265) * Capture Sentry errors prior to initialization Sentry errors captured before/during the wallet initialization are currently not captured because we don't have the controller state yet to determine whether the user has consented. The Sentry setup has been updated to check the persisted state for whether the user has consented, as a fallback in case the controller state hasn't been initialized yet. This ensures that we capture errors during initialization if the user has opted in. * Always await async check for whether the user has opted in * Remove unused import * Update JSDoc return type * Remove unused driver method * Fix metametrics controller unit tests * Fix e2e tests * Fix e2e test on Firefox * Start session upon install rather than toggle --- app/scripts/background.js | 4 + app/scripts/controllers/metametrics.js | 8 +- app/scripts/controllers/metametrics.test.js | 14 +- app/scripts/lib/sentry-filter-events.ts | 8 +- app/scripts/lib/setup-persisted-state-hook.js | 10 + app/scripts/lib/setupSentry.js | 36 +-- app/scripts/sentry-install.js | 7 +- app/scripts/ui.js | 4 + test/e2e/tests/errors.spec.js | 227 ++++++++++++------ 9 files changed, 216 insertions(+), 102 deletions(-) create mode 100644 app/scripts/lib/setup-persisted-state-hook.js diff --git a/app/scripts/background.js b/app/scripts/background.js index d949aefc8394..9dc28d238ae0 100644 --- a/app/scripts/background.js +++ b/app/scripts/background.js @@ -2,6 +2,10 @@ * @file The entry point for the web extension singleton process. */ +// This import sets up a global function required for Sentry to function. +// It must be run first in case an error is thrown later during initialization. +import './lib/setup-persisted-state-hook'; + import EventEmitter from 'events'; import endOfStream from 'end-of-stream'; import pump from 'pump'; diff --git a/app/scripts/controllers/metametrics.js b/app/scripts/controllers/metametrics.js index b5ef191fce9a..20a7cdcb5493 100644 --- a/app/scripts/controllers/metametrics.js +++ b/app/scripts/controllers/metametrics.js @@ -419,18 +419,18 @@ export default class MetaMetricsController { * * @param {boolean} participateInMetaMetrics - Whether or not the user wants * to participate in MetaMetrics - * @returns {string|null} the string of the new metametrics id, or null + * @returns {Promise} the string of the new metametrics id, or null * if not set */ - setParticipateInMetaMetrics(participateInMetaMetrics) { + async setParticipateInMetaMetrics(participateInMetaMetrics) { let { metaMetricsId } = this.state; if (participateInMetaMetrics && !metaMetricsId) { // We also need to start sentry automatic session tracking at this point - globalThis.sentry?.startSession(); + await globalThis.sentry?.startSession(); metaMetricsId = this.generateMetaMetricsId(); } else if (participateInMetaMetrics === false) { // We also need to stop sentry automatic session tracking at this point - globalThis.sentry?.endSession(); + await globalThis.sentry?.endSession(); metaMetricsId = null; } this.store.updateState({ participateInMetaMetrics, metaMetricsId }); diff --git a/app/scripts/controllers/metametrics.test.js b/app/scripts/controllers/metametrics.test.js index ceab26a5a299..09170d4eee8f 100644 --- a/app/scripts/controllers/metametrics.test.js +++ b/app/scripts/controllers/metametrics.test.js @@ -313,30 +313,30 @@ describe('MetaMetricsController', function () { }); describe('setParticipateInMetaMetrics', function () { - it('should update the value of participateInMetaMetrics', function () { + it('should update the value of participateInMetaMetrics', async function () { const metaMetricsController = getMetaMetricsController({ participateInMetaMetrics: null, metaMetricsId: null, }); assert.equal(metaMetricsController.state.participateInMetaMetrics, null); - metaMetricsController.setParticipateInMetaMetrics(true); + await metaMetricsController.setParticipateInMetaMetrics(true); assert.ok(globalThis.sentry.startSession.calledOnce); assert.equal(metaMetricsController.state.participateInMetaMetrics, true); - metaMetricsController.setParticipateInMetaMetrics(false); + await metaMetricsController.setParticipateInMetaMetrics(false); assert.equal(metaMetricsController.state.participateInMetaMetrics, false); }); - it('should generate and update the metaMetricsId when set to true', function () { + it('should generate and update the metaMetricsId when set to true', async function () { const metaMetricsController = getMetaMetricsController({ participateInMetaMetrics: null, metaMetricsId: null, }); assert.equal(metaMetricsController.state.metaMetricsId, null); - metaMetricsController.setParticipateInMetaMetrics(true); + await metaMetricsController.setParticipateInMetaMetrics(true); assert.equal(typeof metaMetricsController.state.metaMetricsId, 'string'); }); - it('should nullify the metaMetricsId when set to false', function () { + it('should nullify the metaMetricsId when set to false', async function () { const metaMetricsController = getMetaMetricsController(); - metaMetricsController.setParticipateInMetaMetrics(false); + await metaMetricsController.setParticipateInMetaMetrics(false); assert.ok(globalThis.sentry.endSession.calledOnce); assert.equal(metaMetricsController.state.metaMetricsId, null); }); diff --git a/app/scripts/lib/sentry-filter-events.ts b/app/scripts/lib/sentry-filter-events.ts index 050f0bcd25e7..50d2f4c77245 100644 --- a/app/scripts/lib/sentry-filter-events.ts +++ b/app/scripts/lib/sentry-filter-events.ts @@ -29,7 +29,7 @@ export class FilterEvents implements Integration { * @returns `true` if MetaMask's state has been initialized, and MetaMetrics * is enabled, `false` otherwise. */ - private getMetaMetricsEnabled: () => boolean; + private getMetaMetricsEnabled: () => Promise; /** * @param options - Constructor options. @@ -40,7 +40,7 @@ export class FilterEvents implements Integration { constructor({ getMetaMetricsEnabled, }: { - getMetaMetricsEnabled: () => boolean; + getMetaMetricsEnabled: () => Promise; }) { this.getMetaMetricsEnabled = getMetaMetricsEnabled; } @@ -56,13 +56,13 @@ export class FilterEvents implements Integration { addGlobalEventProcessor: (callback: EventProcessor) => void, getCurrentHub: () => Hub, ): void { - addGlobalEventProcessor((currentEvent: SentryEvent) => { + addGlobalEventProcessor(async (currentEvent: SentryEvent) => { // Sentry integrations use the Sentry hub to get "this" references, for // reasons I don't fully understand. // eslint-disable-next-line consistent-this const self = getCurrentHub().getIntegration(FilterEvents); if (self) { - if (!self.getMetaMetricsEnabled()) { + if (!(await self.getMetaMetricsEnabled())) { logger.warn(`Event dropped due to MetaMetrics setting.`); return null; } diff --git a/app/scripts/lib/setup-persisted-state-hook.js b/app/scripts/lib/setup-persisted-state-hook.js new file mode 100644 index 000000000000..9b29fad2601a --- /dev/null +++ b/app/scripts/lib/setup-persisted-state-hook.js @@ -0,0 +1,10 @@ +import LocalStore from './local-store'; +import ReadOnlyNetworkStore from './network-store'; + +const localStore = process.env.IN_TEST + ? new ReadOnlyNetworkStore() + : new LocalStore(); + +globalThis.stateHooks.getPersistedState = async function () { + return await localStore.get(); +}; diff --git a/app/scripts/lib/setupSentry.js b/app/scripts/lib/setupSentry.js index 7df0cea78311..8cdc65223a94 100644 --- a/app/scripts/lib/setupSentry.js +++ b/app/scripts/lib/setupSentry.js @@ -118,16 +118,20 @@ export default function setupSentry({ release, getState }) { * @returns `true` if MetaMask's state has been initialized, and MetaMetrics * is enabled, `false` otherwise. */ - function getMetaMetricsEnabled() { - if (getState) { - const appState = getState(); - if (!appState?.store?.metamask?.participateInMetaMetrics) { - return false; - } - } else { + async function getMetaMetricsEnabled() { + const appState = getState(); + if (Object.keys(appState) > 0) { + return Boolean(appState?.store?.metamask?.participateInMetaMetrics); + } + try { + const persistedState = await globalThis.stateHooks.getPersistedState(); + return Boolean( + persistedState?.data?.MetaMetricsController?.participateInMetaMetrics, + ); + } catch (error) { + console.error(error); return false; } - return true; } Sentry.init({ @@ -186,10 +190,10 @@ export default function setupSentry({ release, getState }) { * opted into MetaMetrics, change the autoSessionTracking option and start * a new sentry session. */ - const startSession = () => { + const startSession = async () => { const hub = Sentry.getCurrentHub?.(); const options = hub.getClient?.().getOptions?.() ?? {}; - if (hub && getMetaMetricsEnabled() === true) { + if (hub && (await getMetaMetricsEnabled()) === true) { options.autoSessionTracking = true; hub.startSession(); } @@ -200,10 +204,10 @@ export default function setupSentry({ release, getState }) { * opted out of MetaMetrics, change the autoSessionTracking option and end * the current sentry session. */ - const endSession = () => { + const endSession = async () => { const hub = Sentry.getCurrentHub?.(); const options = hub.getClient?.().getOptions?.() ?? {}; - if (hub && getMetaMetricsEnabled() === false) { + if (hub && (await getMetaMetricsEnabled()) === false) { options.autoSessionTracking = false; hub.endSession(); } @@ -214,22 +218,22 @@ export default function setupSentry({ release, getState }) { * on the state of metaMetrics optin and the state of autoSessionTracking on * the Sentry client. */ - const toggleSession = () => { + const toggleSession = async () => { const hub = Sentry.getCurrentHub?.(); const options = hub.getClient?.().getOptions?.() ?? { autoSessionTracking: false, }; - const isMetaMetricsEnabled = getMetaMetricsEnabled(); + const isMetaMetricsEnabled = await getMetaMetricsEnabled(); if ( isMetaMetricsEnabled === true && options.autoSessionTracking === false ) { - startSession(); + await startSession(); } else if ( isMetaMetricsEnabled === false && options.autoSessionTracking === true ) { - endSession(); + await endSession(); } }; diff --git a/app/scripts/sentry-install.js b/app/scripts/sentry-install.js index aa1264c04a05..99d8ddd39327 100644 --- a/app/scripts/sentry-install.js +++ b/app/scripts/sentry-install.js @@ -10,17 +10,16 @@ global.sentry = setupSentry({ }); /** - * As soon as state is available via getSentryState we can call the - * toggleSession method added to sentry in setupSentry to start automatic + * As soon as state is available via getSentryState we can start automatic * session tracking. */ -function waitForStateHooks() { +async function waitForStateHooks() { if (global.stateHooks.getSentryState) { // sentry is not defined in dev mode, so if sentry doesn't exist at this // point it means that we are in dev mode and do not need to toggle the // session. Using optional chaining is sufficient to prevent the error in // development. - global.sentry?.toggleSession(); + await global.sentry?.startSession(); } else { setTimeout(waitForStateHooks, 100); } diff --git a/app/scripts/ui.js b/app/scripts/ui.js index b3aa15c6ce44..50f37c430044 100644 --- a/app/scripts/ui.js +++ b/app/scripts/ui.js @@ -1,6 +1,10 @@ // dev only, "react-devtools" import is skipped in prod builds import 'react-devtools'; +// This import sets up a global function required for Sentry to function. +// It must be run first in case an error is thrown later during initialization. +import './lib/setup-persisted-state-hook'; + import PortStream from 'extension-port-stream'; import browser from 'webextension-polyfill'; diff --git a/test/e2e/tests/errors.spec.js b/test/e2e/tests/errors.spec.js index 6285ac3b6f30..960135215e56 100644 --- a/test/e2e/tests/errors.spec.js +++ b/test/e2e/tests/errors.spec.js @@ -1,9 +1,26 @@ const { strict: assert } = require('assert'); +const { Browser } = require('selenium-webdriver'); const { convertToHexValue, withFixtures } = require('../helpers'); const FixtureBuilder = require('../fixture-builder'); describe('Sentry errors', function () { - async function mockSentry(mockServer) { + const migrationError = + process.env.SELENIUM_BROWSER === Browser.CHROME + ? `Cannot read properties of undefined (reading 'version')` + : 'meta is undefined'; + async function mockSentryMigratorError(mockServer) { + return await mockServer + .forPost('https://sentry.io/api/0000000/envelope/') + .withBodyIncluding(migrationError) + .thenCallback(() => { + return { + statusCode: 200, + json: {}, + }; + }); + } + + async function mockSentryTestError(mockServer) { return await mockServer .forPost('https://sentry.io/api/0000000/envelope/') .withBodyIncluding('Test Error') @@ -23,73 +40,149 @@ describe('Sentry errors', function () { }, ], }; - it('should NOT send error events when participateInMetaMetrics is false', async function () { - await withFixtures( - { - fixtures: new FixtureBuilder() - .withMetaMetricsController({ - metaMetricsId: null, - participateInMetaMetrics: false, - }) - .build(), - ganacheOptions, - title: this.test.title, - failOnConsoleError: false, - testSpecificMock: mockSentry, - }, - async ({ driver, mockedEndpoint }) => { - await driver.navigate(); - await driver.fill('#password', 'correct horse battery staple'); - await driver.press('#password', driver.Key.ENTER); - // Trigger error - driver.executeScript('window.stateHooks.throwTestError()'); - driver.delay(3000); - // Wait for Sentry request - const isPending = await mockedEndpoint.isPending(); - assert.ok( - isPending, - 'A request to sentry was sent when it should not have been', - ); - }, - ); + + describe('before initialization', function () { + it('should NOT send error events when participateInMetaMetrics is false', async function () { + await withFixtures( + { + fixtures: { + ...new FixtureBuilder() + .withMetaMetricsController({ + metaMetricsId: null, + participateInMetaMetrics: false, + }) + .build(), + // Intentionally corrupt state to trigger migration error during initialization + meta: undefined, + }, + ganacheOptions, + title: this.test.title, + failOnConsoleError: false, + testSpecificMock: mockSentryMigratorError, + }, + async ({ driver, mockedEndpoint }) => { + await driver.navigate(); + + // Wait for Sentry request + await driver.delay(3000); + const isPending = await mockedEndpoint.isPending(); + assert.ok( + isPending, + 'A request to sentry was sent when it should not have been', + ); + }, + ); + }); + it('should send error events', async function () { + await withFixtures( + { + fixtures: { + ...new FixtureBuilder() + .withMetaMetricsController({ + metaMetricsId: 'fake-metrics-id', + participateInMetaMetrics: true, + }) + .build(), + // Intentionally corrupt state to trigger migration error during initialization + meta: undefined, + }, + ganacheOptions, + title: this.test.title, + failOnConsoleError: false, + testSpecificMock: mockSentryMigratorError, + }, + async ({ driver, mockedEndpoint }) => { + await driver.navigate(); + + // Wait for Sentry request + await driver.wait(async () => { + const isPending = await mockedEndpoint.isPending(); + return isPending === false; + }, 3000); + + const [mockedRequest] = await mockedEndpoint.getSeenRequests(); + const mockTextBody = mockedRequest.body.text.split('\n'); + const mockJsonBody = JSON.parse(mockTextBody[2]); + const { level } = mockJsonBody; + const [{ type, value }] = mockJsonBody.exception.values; + // Verify request + assert.equal(type, 'TypeError'); + assert(value.includes(migrationError)); + assert.equal(level, 'error'); + }, + ); + }); }); - it('should send error events', async function () { - await withFixtures( - { - fixtures: new FixtureBuilder() - .withMetaMetricsController({ - metaMetricsId: 'fake-metrics-id', - participateInMetaMetrics: true, - }) - .build(), - ganacheOptions, - title: this.test.title, - failOnConsoleError: false, - testSpecificMock: mockSentry, - }, - async ({ driver, mockedEndpoint }) => { - await driver.navigate(); - await driver.fill('#password', 'correct horse battery staple'); - await driver.press('#password', driver.Key.ENTER); - // Trigger error - driver.executeScript('window.stateHooks.throwTestError()'); - // Wait for Sentry request - await driver.wait(async () => { + + describe('after initialization', function () { + it('should NOT send error events when participateInMetaMetrics is false', async function () { + await withFixtures( + { + fixtures: new FixtureBuilder() + .withMetaMetricsController({ + metaMetricsId: null, + participateInMetaMetrics: false, + }) + .build(), + ganacheOptions, + title: this.test.title, + failOnConsoleError: false, + testSpecificMock: mockSentryTestError, + }, + async ({ driver, mockedEndpoint }) => { + await driver.navigate(); + await driver.fill('#password', 'correct horse battery staple'); + await driver.press('#password', driver.Key.ENTER); + // Trigger error + driver.executeScript('window.stateHooks.throwTestError()'); + driver.delay(3000); + // Wait for Sentry request const isPending = await mockedEndpoint.isPending(); - return isPending === false; - }, 10000); - const [mockedRequest] = await mockedEndpoint.getSeenRequests(); - const mockTextBody = mockedRequest.body.text.split('\n'); - const mockJsonBody = JSON.parse(mockTextBody[2]); - const { level, extra } = mockJsonBody; - const [{ type, value }] = mockJsonBody.exception.values; - const { participateInMetaMetrics } = extra.appState.store.metamask; - // Verify request - assert.equal(type, 'TestError'); - assert.equal(value, 'Test Error'); - assert.equal(level, 'error'); - assert.equal(participateInMetaMetrics, true); - }, - ); + assert.ok( + isPending, + 'A request to sentry was sent when it should not have been', + ); + }, + ); + }); + it('should send error events', async function () { + await withFixtures( + { + fixtures: new FixtureBuilder() + .withMetaMetricsController({ + metaMetricsId: 'fake-metrics-id', + participateInMetaMetrics: true, + }) + .build(), + ganacheOptions, + title: this.test.title, + failOnConsoleError: false, + testSpecificMock: mockSentryTestError, + }, + async ({ driver, mockedEndpoint }) => { + await driver.navigate(); + await driver.fill('#password', 'correct horse battery staple'); + await driver.press('#password', driver.Key.ENTER); + // Trigger error + driver.executeScript('window.stateHooks.throwTestError()'); + // Wait for Sentry request + await driver.wait(async () => { + const isPending = await mockedEndpoint.isPending(); + return isPending === false; + }, 10000); + const [mockedRequest] = await mockedEndpoint.getSeenRequests(); + const mockTextBody = mockedRequest.body.text.split('\n'); + const mockJsonBody = JSON.parse(mockTextBody[2]); + const { level, extra } = mockJsonBody; + const [{ type, value }] = mockJsonBody.exception.values; + const { participateInMetaMetrics } = extra.appState.store.metamask; + // Verify request + assert.equal(type, 'TestError'); + assert.equal(value, 'Test Error'); + assert.equal(level, 'error'); + assert.equal(participateInMetaMetrics, true); + }, + ); + }); }); }); From 3b0d37c3bf95e770ff69b54fcba58817d847fd67 Mon Sep 17 00:00:00 2001 From: Dan J Miller Date: Mon, 31 Jul 2023 19:43:51 -0230 Subject: [PATCH 23/25] Fix migration 77 (#20276) * Handle the case where tokensChainsCache data is undefined in migration 77 * Delete parts of state that should have been removed in migrations 82,84,86 and 88 * Create 077-supplements.md * Update 077-supplements.md * Update 077-supplements/*.js code comments * Fix types and jsdoc * Type/lint fix * Cleanup * Add 'should set data to an empty object if it is null' test case to 077.test.js * Update app/scripts/migrations/077.test.js Co-authored-by: Mark Stacey * Modify deletion criteria so that all decimal chain id proprties are deleted in migration 88 supplement * Readme.md * Update app/scripts/migrations/077.test.js Co-authored-by: Mark Stacey * Update app/scripts/migrations/077.test.js Co-authored-by: Mark Stacey * Update app/scripts/migrations/077.test.js Co-authored-by: Mark Stacey * Lint fix * Only delete decimal chain id keyed-entries in migration 88 supplement if there are hexadecimal keyed entries as well * Remove redundant test * Add 'does not delete' cases for nftcontroller related tests in 077.test.js --------- Co-authored-by: Mark Stacey --- .../077-supplements/077-supplement-for-082.ts | 25 + .../077-supplements/077-supplement-for-084.ts | 24 + .../077-supplements/077-supplement-for-086.ts | 23 + .../077-supplements/077-supplement-for-088.ts | 152 ++ .../077-supplements/077-supplements.md | 100 ++ app/scripts/migrations/077.js | 18 +- app/scripts/migrations/077.test.js | 1247 +++++++++++++++++ 7 files changed, 1585 insertions(+), 4 deletions(-) create mode 100644 app/scripts/migrations/077-supplements/077-supplement-for-082.ts create mode 100644 app/scripts/migrations/077-supplements/077-supplement-for-084.ts create mode 100644 app/scripts/migrations/077-supplements/077-supplement-for-086.ts create mode 100644 app/scripts/migrations/077-supplements/077-supplement-for-088.ts create mode 100644 app/scripts/migrations/077-supplements/077-supplements.md diff --git a/app/scripts/migrations/077-supplements/077-supplement-for-082.ts b/app/scripts/migrations/077-supplements/077-supplement-for-082.ts new file mode 100644 index 000000000000..149b2ab0c226 --- /dev/null +++ b/app/scripts/migrations/077-supplements/077-supplement-for-082.ts @@ -0,0 +1,25 @@ +import { hasProperty, isObject } from '@metamask/utils'; + +/** + * Deletes frequentRpcListDetail if networkConfigurations exists, on the NetworkController state. + * Further explanation in ./077-supplements.md + * + * @param state - The persisted MetaMask state, keyed by controller. + * @returns Updated versioned MetaMask extension state. + */ + +export default function transformState077For082( + state: Record, +) { + if ( + hasProperty(state, 'PreferencesController') && + isObject(state.PreferencesController) && + hasProperty(state.PreferencesController, 'frequentRpcListDetail') && + isObject(state.NetworkController) && + hasProperty(state.NetworkController, 'networkConfigurations') + ) { + delete state.PreferencesController.frequentRpcListDetail; + } + + return { ...state }; +} diff --git a/app/scripts/migrations/077-supplements/077-supplement-for-084.ts b/app/scripts/migrations/077-supplements/077-supplement-for-084.ts new file mode 100644 index 000000000000..397efec01b9d --- /dev/null +++ b/app/scripts/migrations/077-supplements/077-supplement-for-084.ts @@ -0,0 +1,24 @@ +import { hasProperty, isObject } from '@metamask/utils'; + +/** + * Deletes network if networkId exists, on the NetworkController state. + * Further explanation in ./077-supplements.md + * + * @param state - The persisted MetaMask state, keyed by controller. + * @returns Updated versioned MetaMask extension state. + */ + +export default function transformState077For084( + state: Record, +) { + if ( + hasProperty(state, 'NetworkController') && + isObject(state.NetworkController) && + hasProperty(state.NetworkController, 'network') && + hasProperty(state.NetworkController, 'networkId') + ) { + delete state.NetworkController.network; + } + + return { ...state }; +} diff --git a/app/scripts/migrations/077-supplements/077-supplement-for-086.ts b/app/scripts/migrations/077-supplements/077-supplement-for-086.ts new file mode 100644 index 000000000000..bad44820e7b3 --- /dev/null +++ b/app/scripts/migrations/077-supplements/077-supplement-for-086.ts @@ -0,0 +1,23 @@ +import { hasProperty, isObject } from '@metamask/utils'; + +/** + * Prior to token detection v2 the data property in tokensChainsCache was an array, + * in v2 we changes that to an object. In this migration we are converting the data as array to object. + * + * @param state - The persisted MetaMask state, keyed by controller. + * @returns Updated versioned MetaMask extension state. + */ +export default function transformState077For086( + state: Record, +) { + if ( + hasProperty(state, 'NetworkController') && + isObject(state.NetworkController) && + hasProperty(state.NetworkController, 'provider') && + hasProperty(state.NetworkController, 'providerConfig') + ) { + delete state.NetworkController.provider; + } + + return { ...state }; +} diff --git a/app/scripts/migrations/077-supplements/077-supplement-for-088.ts b/app/scripts/migrations/077-supplements/077-supplement-for-088.ts new file mode 100644 index 000000000000..4d430865ab54 --- /dev/null +++ b/app/scripts/migrations/077-supplements/077-supplement-for-088.ts @@ -0,0 +1,152 @@ +import { hasProperty, isObject, isStrictHexString } from '@metamask/utils'; + +/** + * Deletes properties of `NftController.allNftContracts`, `NftController.allNfts`, + * `TokenListController.tokensChainsCache`, `TokensController.allTokens`, + * `TokensController.allIgnoredTokens` and `TokensController.allDetectedTokens` if + * their keyed by decimal number chainId and another hexadecimal chainId property + * exists within the same object. + * Further explanation in ./077-supplements.md + * + * @param state - The persisted MetaMask state, keyed by controller. + * @returns Updated versioned MetaMask extension state. + */ +export default function transformState077For086( + state: Record, +): Record { + if (hasProperty(state, 'NftController') && isObject(state.NftController)) { + const nftControllerState = state.NftController; + + // Migrate NftController.allNftContracts + if ( + hasProperty(nftControllerState, 'allNftContracts') && + isObject(nftControllerState.allNftContracts) + ) { + const { allNftContracts } = nftControllerState; + + if ( + Object.keys(allNftContracts).every((address) => + isObject(allNftContracts[address]), + ) + ) { + Object.keys(allNftContracts).forEach((address) => { + const nftContractsByChainId = allNftContracts[address]; + if ( + isObject(nftContractsByChainId) && + anyKeysAreHex(nftContractsByChainId) + ) { + for (const chainId of Object.keys(nftContractsByChainId)) { + if (!isStrictHexString(chainId)) { + delete nftContractsByChainId[chainId]; + } + } + } + }); + } + } + + // Migrate NftController.allNfts + if ( + hasProperty(nftControllerState, 'allNfts') && + isObject(nftControllerState.allNfts) + ) { + const { allNfts } = nftControllerState; + + if (Object.keys(allNfts).every((address) => isObject(allNfts[address]))) { + Object.keys(allNfts).forEach((address) => { + const nftsByChainId = allNfts[address]; + if (isObject(nftsByChainId) && anyKeysAreHex(nftsByChainId)) { + for (const chainId of Object.keys(nftsByChainId)) { + if (!isStrictHexString(chainId)) { + delete nftsByChainId[chainId]; + } + } + } + }); + } + } + + state.NftController = nftControllerState; + } + + if ( + hasProperty(state, 'TokenListController') && + isObject(state.TokenListController) + ) { + const tokenListControllerState = state.TokenListController; + + // Migrate TokenListController.tokensChainsCache + if ( + hasProperty(tokenListControllerState, 'tokensChainsCache') && + isObject(tokenListControllerState.tokensChainsCache) && + anyKeysAreHex(tokenListControllerState.tokensChainsCache) + ) { + for (const chainId of Object.keys( + tokenListControllerState.tokensChainsCache, + )) { + if (!isStrictHexString(chainId)) { + delete tokenListControllerState.tokensChainsCache[chainId]; + } + } + } + } + + if ( + hasProperty(state, 'TokensController') && + isObject(state.TokensController) + ) { + const tokensControllerState = state.TokensController; + + // Migrate TokensController.allTokens + if ( + hasProperty(tokensControllerState, 'allTokens') && + isObject(tokensControllerState.allTokens) && + anyKeysAreHex(tokensControllerState.allTokens) + ) { + const { allTokens } = tokensControllerState; + + for (const chainId of Object.keys(allTokens)) { + if (!isStrictHexString(chainId)) { + delete tokensControllerState.allTokens[chainId]; + } + } + } + + // Migrate TokensController.allIgnoredTokens + if ( + hasProperty(tokensControllerState, 'allIgnoredTokens') && + isObject(tokensControllerState.allIgnoredTokens) && + anyKeysAreHex(tokensControllerState.allIgnoredTokens) + ) { + const { allIgnoredTokens } = tokensControllerState; + + for (const chainId of Object.keys(allIgnoredTokens)) { + if (!isStrictHexString(chainId)) { + delete tokensControllerState.allIgnoredTokens[chainId]; + } + } + } + + // Migrate TokensController.allDetectedTokens + if ( + hasProperty(tokensControllerState, 'allDetectedTokens') && + isObject(tokensControllerState.allDetectedTokens) && + anyKeysAreHex(tokensControllerState.allDetectedTokens) + ) { + const { allDetectedTokens } = tokensControllerState; + + for (const chainId of Object.keys(allDetectedTokens)) { + if (!isStrictHexString(chainId)) { + delete tokensControllerState.allDetectedTokens[chainId]; + } + } + } + + state.TokensController = tokensControllerState; + } + return state; +} + +function anyKeysAreHex(obj: Record) { + return Object.keys(obj).some((chainId) => isStrictHexString(chainId)); +} diff --git a/app/scripts/migrations/077-supplements/077-supplements.md b/app/scripts/migrations/077-supplements/077-supplements.md new file mode 100644 index 000000000000..ca40d9852066 --- /dev/null +++ b/app/scripts/migrations/077-supplements/077-supplements.md @@ -0,0 +1,100 @@ +# 077 Supplements + +As of the time this file was first PR'd, we had not yet had to do what was done in this PR, which is to fix an old migration and also supplement it with state transformations +to handle problems that could be introduced by future migrations. + +The document explains the need for these new state transformations and the rationale behind each. It also explains why other state transformations were not included. + +## Background + +As of release 10.34.0, we started having a `No metadata found for 'previousProviderStore'` error thrown from the `deriveStateFromMetadata` function in `BaseControllerV2.js`. +This was occuring when there was data on the NetworkController state for which the NetworkController + BaseController expect metadata, but no metadata exists. In particular, +`previousProviderStore` was on the NetworkController state when it should not have been. + +`previousProviderStore` should not have been on the NetworkController state because of migration 85, which explictly deletes it. + +We discovered that for some users, that migration had failed to run because of an error in an earlier migration: `TypeError#1: MetaMask Migration Error #77: Cannot convert undefined or null to object`. +This error was thrown from this line /~https://github.com/MetaMask/metamask-extension/commit/8f18e04b97af02e5a8a72e3e4872aac66595d1d8#diff-9e76a7c60c1e37cd949f729222338b23ab743e44938ccf63a4a6dab7d84ed8bcR38 + +So the `data` property of the objects within `TokenListController.tokensChainsCache` could be undefined, and migration 77 didn't handle that case. It could be undefined because of the way the assets controller +code was as of the core controller libraries 14.0.2 release /~https://github.com/MetaMask/core/blame/19f7bf3b9fd8abe6cef9cb1ac1fe831d9f651ae0/src/assets/TokenListController.ts#L270 (the `safelyExecute` call there +will return undefined if the network call failed) + +For users who were in that situation, where a `TokenListController.tokensChainsCache[chainId].data` property was undefined, some significant problems would occur after updating to v10.24.0, which is the +release where migration 77 was shipped. In particular, migration 77 would fail, and all subsequent migrations would not run. The most plain case of this would be a user who was on v10.23.0 +with `TokenListController.tokensChainsCache[chainId].data === undefined`. Then suppose they didn't update until v10.34.0. None of migrations 77-89 would run. Leaving their state in a form that doesn't match +with assumptions throughout our codebase. Various bugs could be caused, but the sentry error describe above is the worst case, where MetaMask simply could not be opened and users would hit the error screen. + +To correct this situation we had to fix migration 77. Once we do that, all users who were in this situation (and then upgraded to the version which included the fixes for migration 77) would have all migrations +from 77 upwards run for the first time. This could be problematic for users who had used the wallet on versions 10.24.0-10.34.0, where our controllers would be writing data to state under the assumption that +the migrations had run. + +As such, we had to also add code to migration 77 to avoid new errors being introduced by the migrations running on code that had been modified by controllers on versions 10.24.0 to 10.34.0 + +## Introducing migration 77 supplements + +To correct the aforementioned problems with the data, new state transformation functions were added to this directory, to be run in migration 77 after the pre-existing migration 77 code had run. +Each file in this directory exports a state transformation function which is then imported in the migration 77 file and applied to state in sequence, after the state transformation function in +077.js itself has run and returns state. These have been split into their own files for each of use, and so that they could be grouped with this documentation. + +## The chosen supplements + +We added supplements for migrations 82, 84, 86 and 88 for the following reasons and with the following effects -> + +**Migration 82** + +Attempts to convert `frequentRpcListDetail` - an array of objects with network related data - to a `networkConfigurations` object, with the objects that were in the array keyed by a unique id. +If this migration had not run, then (prior to v10.34.0) a user would still have been able to use MetaMask, but the data they had in `frequentRpcListDetail` would now be ignored by application code, +and subsequent network data would between written to and modified in state in the `networkConfigurations` object. If migration 82 was later run (after fixing migration 77), the old `frequentRpcListDetail` +data could overwrite the existing `networkConfigurations` data, and the user could lose `networkConfigurations` data that had been written to their state since migration 82 had first failed to run. + +To fix this, the migration 82 supplement deletes `frequentRpcListDetail` if the `networkConfigurations` object exists. Users in such a scenario will have network data in `networkConfigurations` that +they have been using, while the `frequentRpcListDetail` data would not have been seen for some time. So the best thing to do for them is delete their old data and preserve the data they have most recently +used. + +**Migration 84** + +Replaces the `NetworkController.network` property with a `networkId` and `networkStatus` property. If this migration had not run, the NetworkController would have a `network` property and +`networkId` and `networkStatus` properties. If migration 84 later ran on this state, the old (and forgotten) `network` property could cause the `networkId` and `networkStatus` to be overwritten, +affecting the state the user's experience was currently depending on. + +The fix in the migration 84 supplement is to delete the `NetworkController.network` property if the `NetworkId` property already exists. + +**Migration 86** + +Renamed the `NetworkController.provider` property to `providerConfig`. If this migration had not run, the NetworkController would have a `provider` property and +a `providerConfig` property. If migration 86 later ran on this state, the old (and forgotten) `provider` property could cause the `providerConfig` property to be overwritten, +affecting the state the user's experience was currently depending on. + +The fix in the migration 86 supplement is to delete the `NetworkController.provider` property if the `providerConfig` property already exists. + +**Migration 88** + +Attempted to change the keys of multiple parts of state related to tokens. In particular, `NftController.allNftContracts`, `NftController.allNfts`, `TokenListController.tokensChainsCache`, `TokensController.allTokens`, `TokensController.allIgnoredTokens` and `TokensController.allDetectedTokens`. All of these objects were keyed by chainId in decimal number form. The migration's +purpose was to change those decimal chain ID keys to hexadecimal. If migration 77 failed, and then the user added or modified tokens, they could have duplicates within these parts of state: +some with decimal keys and others with an equivalent hexadecimal key. If the data pointed to by those keys was modified at all, and the migration 88 was later run, the most recent data (under +the hexadecimal key) could be overwritten by the old data under the decimal key. + +The migration 88 supplement fixes this by deleting the properties with decimal keys if an equivalent hexadecimal key exists. + +## Migrations that were not supplemented + +**Migration 78** was not supplemented because it only deletes data; it does not overwrite data. It's failure to run will have left rogue data in state, but that will be removed when it is run after the migration +77 fix. + +**Migration 79** was not supplemented because it only deletes data; it does not overwrite data. + +**Migration 80** was not supplemented because it only deletes data; it does not overwrite data. + +**Migration 81** was not supplemented because it modifies data that could only be in state on a flask build. The bug that caused the undefined data in tokenlistcontroller state was present on v14.0.2 and v14.1.0 of +the controllers, but fixed in v14.2.0 of the controllers. By the time flask was released to prod, controllers was at v25.0 + +**Migration 83** just builds on migration 82. No additional fix is needed for 83 given that we have the 82 supplement. + +**Migration 85** was not supplemented because it only deletes data; it does not overwrite data. + +**Migration 87** was not supplemented because it only deletes data; it does not overwrite data. + +**Migration 89** just builds on migration 82 and 84. No additional fix is needed for 89 given that we have the 82 and 84 supplement. + +**Migration 90** was not supplemented because it only deletes data; it does not overwrite data. diff --git a/app/scripts/migrations/077.js b/app/scripts/migrations/077.js index 141cbb142db8..5a5d4f1ac2c9 100644 --- a/app/scripts/migrations/077.js +++ b/app/scripts/migrations/077.js @@ -1,4 +1,8 @@ import { cloneDeep } from 'lodash'; +import transformState077For082 from './077-supplements/077-supplement-for-082'; +import transformState077For084 from './077-supplements/077-supplement-for-084'; +import transformState077For086 from './077-supplements/077-supplement-for-086'; +import transformState077For088 from './077-supplements/077-supplement-for-088'; const version = 77; @@ -12,7 +16,13 @@ export default { const versionedData = cloneDeep(originalVersionedData); versionedData.meta.version = version; const state = versionedData.data; - const newState = transformState(state); + let newState = transformState(state); + + newState = transformState077For082(newState); + newState = transformState077For084(newState); + newState = transformState077For086(newState); + newState = transformState077For088(newState); + versionedData.data = newState; return versionedData; }, @@ -27,7 +37,7 @@ function transformState(state) { let dataObject; // eslint-disable-next-line for (const chainId in tokensChainsCache) { - dataCache = tokensChainsCache[chainId].data; + dataCache = tokensChainsCache[chainId].data || {}; dataObject = {}; // if the data is array conver that to object if (Array.isArray(dataCache)) { @@ -35,8 +45,8 @@ function transformState(state) { dataObject[token.address] = token; } } else if ( - Object.keys(dataCache)[0].toLowerCase() !== - dataCache[Object.keys(dataCache)[0]].address.toLowerCase() + Object.keys(dataCache)[0]?.toLowerCase() !== + dataCache[Object.keys(dataCache)[0]]?.address?.toLowerCase() ) { // for the users who already updated to the recent version // and the dataCache is already an object keyed with 0,1,2,3 etc diff --git a/app/scripts/migrations/077.test.js b/app/scripts/migrations/077.test.js index 1c16420ecaed..53efb5cd5f65 100644 --- a/app/scripts/migrations/077.test.js +++ b/app/scripts/migrations/077.test.js @@ -82,6 +82,100 @@ describe('migration #77', () => { }, }); }); + it('should set data to an empty object if it is undefined', async () => { + const oldStorage = { + meta: { + version: 76, + }, + data: { + TokenListController: { + tokenList: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + }, + tokensChainsCache: { + 1: { + timestamp: 1234, + data: undefined, + }, + }, + }, + }, + }; + const newStorage = await migration77.migrate(oldStorage); + expect(newStorage).toStrictEqual({ + meta: { + version: 77, + }, + data: { + TokenListController: { + tokenList: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + }, + tokensChainsCache: { + 1: { + timestamp: 1234, + data: {}, + }, + }, + }, + }, + }); + }); + it('should set data to an empty object if it is null', async () => { + const oldStorage = { + meta: { + version: 76, + }, + data: { + TokenListController: { + tokenList: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + }, + tokensChainsCache: { + 1: { + timestamp: 1234, + data: null, + }, + }, + }, + }, + }; + const newStorage = await migration77.migrate(oldStorage); + expect(newStorage).toStrictEqual({ + meta: { + version: 77, + }, + data: { + TokenListController: { + tokenList: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + }, + tokensChainsCache: { + 1: { + timestamp: 1234, + data: {}, + }, + }, + }, + }, + }); + }); it('should change the data from array to object for a multiple networks', async () => { const oldStorage = { meta: { @@ -319,4 +413,1157 @@ describe('migration #77', () => { }, }); }); + + describe('migration #77 supplements', () => { + describe('state transformation to ahead of migration 82', () => { + it('should delete frequentRpcListDetail from the PreferencesController state, if the user already has networkConfigurations in NetworkController state, without interferring with the rest of the migration', async () => { + const oldStorage = { + meta: { + version: 76, + }, + data: { + TokenListController: { + tokenList: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + }, + tokensChainsCache: { + 1: { + timestamp: 1234, + data: [ + { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + { + address: '0xc00e94cb662c3520282e6f5717214004a7f26888', + symbol: 'COMP', + decimals: 18, + }, + ], + }, + }, + }, + PreferencesController: { + frequentRpcListDetail: ['foobar'], + fizz: 'buzz', + }, + NetworkController: { + networkConfigurations: { foo: 'bar' }, + }, + }, + }; + const newStorage = await migration77.migrate(oldStorage); + expect(newStorage).toStrictEqual({ + meta: { + version: 77, + }, + data: { + TokenListController: { + tokenList: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + }, + tokensChainsCache: { + 1: { + timestamp: 1234, + data: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + '0xc00e94cb662c3520282e6f5717214004a7f26888': { + address: '0xc00e94cb662c3520282e6f5717214004a7f26888', + symbol: 'COMP', + decimals: 18, + }, + }, + }, + }, + }, + PreferencesController: { + fizz: 'buzz', + }, + NetworkController: { + networkConfigurations: { foo: 'bar' }, + }, + }, + }); + }); + + it('should not delete frequentRpcListDetail from the PreferencesController state if there are no networkConfigurations in NetworkController state', async () => { + const oldStorage = { + meta: { + version: 76, + }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + PreferencesController: { + frequentRpcListDetail: ['foobar'], + fizz: 'buzz', + }, + NetworkController: { + foobar: { foo: 'bar' }, + }, + }, + }; + const newStorage = await migration77.migrate(oldStorage); + expect(newStorage).toStrictEqual({ + meta: { + version: 77, + }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + PreferencesController: { + frequentRpcListDetail: ['foobar'], + fizz: 'buzz', + }, + NetworkController: { + foobar: { foo: 'bar' }, + }, + }, + }); + }); + }); + + describe('state transformation to ahead of migration 84', () => { + it('should delete `network` from the NetworkController state, if the user already has `networkId` in NetworkController state, without interferring with the rest of the migration', async () => { + const oldStorage = { + meta: { + version: 76, + }, + data: { + TokenListController: { + tokenList: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + }, + tokensChainsCache: { + 1: { + timestamp: 1234, + data: [ + { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + { + address: '0xc00e94cb662c3520282e6f5717214004a7f26888', + symbol: 'COMP', + decimals: 18, + }, + ], + }, + }, + }, + NetworkController: { + network: 'foobar', + networkId: 'fizzbuzz', + }, + }, + }; + const newStorage = await migration77.migrate(oldStorage); + expect(newStorage).toStrictEqual({ + meta: { + version: 77, + }, + data: { + TokenListController: { + tokenList: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + }, + tokensChainsCache: { + 1: { + timestamp: 1234, + data: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + '0xc00e94cb662c3520282e6f5717214004a7f26888': { + address: '0xc00e94cb662c3520282e6f5717214004a7f26888', + symbol: 'COMP', + decimals: 18, + }, + }, + }, + }, + }, + NetworkController: { + networkId: 'fizzbuzz', + }, + }, + }); + }); + + it('should not delete `network` from the NetworkController state, if there is no `networkId` in NetworkController state', async () => { + const oldStorage = { + meta: { + version: 76, + }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + PreferencesController: { + frequentRpcListDetail: ['foobar'], + fizz: 'buzz', + }, + NetworkController: { + network: 'foobar', + foobar: { foo: 'bar' }, + }, + }, + }; + const newStorage = await migration77.migrate(oldStorage); + expect(newStorage).toStrictEqual({ + meta: { + version: 77, + }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + PreferencesController: { + frequentRpcListDetail: ['foobar'], + fizz: 'buzz', + }, + NetworkController: { + network: 'foobar', + foobar: { foo: 'bar' }, + }, + }, + }); + }); + }); + + describe('state transformation to ahead of migration 86', () => { + it('should delete `provider` from the NetworkController state, if the user already has `providerConfig` in NetworkController state, without interferring with the rest of the migration', async () => { + const oldStorage = { + meta: { + version: 76, + }, + data: { + TokenListController: { + tokenList: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + }, + tokensChainsCache: { + 1: { + timestamp: 1234, + data: [ + { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + { + address: '0xc00e94cb662c3520282e6f5717214004a7f26888', + symbol: 'COMP', + decimals: 18, + }, + ], + }, + }, + }, + NetworkController: { + provider: { foo: 'bar ' }, + providerConfig: { fizz: 'buzz' }, + }, + }, + }; + const newStorage = await migration77.migrate(oldStorage); + expect(newStorage).toStrictEqual({ + meta: { + version: 77, + }, + data: { + TokenListController: { + tokenList: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + }, + tokensChainsCache: { + 1: { + timestamp: 1234, + data: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + '0xc00e94cb662c3520282e6f5717214004a7f26888': { + address: '0xc00e94cb662c3520282e6f5717214004a7f26888', + symbol: 'COMP', + decimals: 18, + }, + }, + }, + }, + }, + NetworkController: { + providerConfig: { fizz: 'buzz' }, + }, + }, + }); + }); + + it('should not delete `provider` from the NetworkController state, if there is no `providerConfig` in NetworkController state', async () => { + const oldStorage = { + meta: { + version: 76, + }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + PreferencesController: { + frequentRpcListDetail: ['foobar'], + fizz: 'buzz', + }, + NetworkController: { + provider: { foo: 'bar ' }, + }, + }, + }; + const newStorage = await migration77.migrate(oldStorage); + expect(newStorage).toStrictEqual({ + meta: { + version: 77, + }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + PreferencesController: { + frequentRpcListDetail: ['foobar'], + fizz: 'buzz', + }, + NetworkController: { + provider: { foo: 'bar ' }, + }, + }, + }); + }); + }); + + describe('state transformation to ahead of migration 88', () => { + it('deletes entries in NftController.allNftContracts that have decimal chain ID keys only if any chain ID keys are hex', async () => { + const oldStorage = { + meta: { version: 76 }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + NftController: { + allNftContracts: { + '0x111': { + 16: [ + { + name: 'Contract 1', + address: '0xaaa', + }, + ], + '0x20': [ + { + name: 'Contract 2', + address: '0xbbb', + }, + ], + 32: [ + { + name: 'Contract 2', + address: '0xbbb', + }, + ], + }, + '0x222': { + 64: [ + { + name: 'Contract 3', + address: '0xccc', + }, + ], + '0x40': [ + { + name: 'Contract 3', + address: '0xccc', + }, + ], + 128: [ + { + name: 'Contract 4', + address: '0xddd', + }, + ], + }, + }, + }, + }, + }; + + const newStorage = await migration77.migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + TokenListController: { + tokensChainsCache: {}, + }, + NftController: { + allNftContracts: { + '0x111': { + '0x20': [ + { + name: 'Contract 2', + address: '0xbbb', + }, + ], + }, + '0x222': { + '0x40': [ + { + name: 'Contract 3', + address: '0xccc', + }, + ], + }, + }, + }, + }); + }); + + it('does not delete entries in NftController.allNftContracts that have decimal chain ID keys if no other chain ID keys are hex', async () => { + const oldStorage = { + meta: { version: 76 }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + NftController: { + allNftContracts: { + '0x333': { + 256: [ + { + name: 'Contract 3', + address: '0xccc', + }, + ], + }, + }, + }, + }, + }; + + const newStorage = await migration77.migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + TokenListController: { + tokensChainsCache: {}, + }, + NftController: { + allNftContracts: { + '0x333': { + 256: [ + { + name: 'Contract 3', + address: '0xccc', + }, + ], + }, + }, + }, + }); + }); + + it('deletes entries in NftController.allNfts that have decimal chain ID keys only if any chain ID keys are hex', async () => { + const oldStorage = { + meta: { version: 76 }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + NftController: { + allNfts: { + '0x111': { + 16: [ + { + name: 'NFT 1', + description: 'Description for NFT 1', + image: 'nft1.jpg', + standard: 'ERC721', + tokenId: '1', + address: '0xaaa', + }, + ], + 32: [ + { + name: 'NFT 2', + description: 'Description for NFT 2', + image: 'nft2.jpg', + standard: 'ERC721', + tokenId: '2', + address: '0xbbb', + }, + ], + '0x20': [ + { + name: 'NFT 2', + description: 'Description for NFT 2', + image: 'nft2.jpg', + standard: 'ERC721', + tokenId: '2', + address: '0xbbb', + }, + ], + }, + '0x222': { + 64: [ + { + name: 'NFT 3', + description: 'Description for NFT 3', + image: 'nft3.jpg', + standard: 'ERC721', + tokenId: '3', + address: '0xccc', + }, + ], + '0x40': [ + { + name: 'NFT 3', + description: 'Description for NFT 3', + image: 'nft3.jpg', + standard: 'ERC721', + tokenId: '3', + address: '0xccc', + }, + ], + 128: [ + { + name: 'NFT 4', + description: 'Description for NFT 4', + image: 'nft4.jpg', + standard: 'ERC721', + tokenId: '4', + address: '0xddd', + }, + ], + }, + '0x333': { + 256: [ + { + name: 'NFT 3', + description: 'Description for NFT 3', + image: 'nft3.jpg', + standard: 'ERC721', + tokenId: '3', + address: '0xccc', + }, + ], + }, + }, + }, + }, + }; + + const newStorage = await migration77.migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + TokenListController: { + tokensChainsCache: {}, + }, + NftController: { + allNfts: { + '0x111': { + '0x20': [ + { + name: 'NFT 2', + description: 'Description for NFT 2', + image: 'nft2.jpg', + standard: 'ERC721', + tokenId: '2', + address: '0xbbb', + }, + ], + }, + '0x222': { + '0x40': [ + { + name: 'NFT 3', + description: 'Description for NFT 3', + image: 'nft3.jpg', + standard: 'ERC721', + tokenId: '3', + address: '0xccc', + }, + ], + }, + '0x333': { + 256: [ + { + name: 'NFT 3', + description: 'Description for NFT 3', + image: 'nft3.jpg', + standard: 'ERC721', + tokenId: '3', + address: '0xccc', + }, + ], + }, + }, + }, + }); + }); + + it('does not delete entries in NftController.allNfts that have decimal chain ID keys if no other chain ID keys are hex', async () => { + const oldStorage = { + meta: { version: 76 }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + NftController: { + allNfts: { + '0x333': { + 256: [ + { + name: 'NFT 3', + description: 'Description for NFT 3', + image: 'nft3.jpg', + standard: 'ERC721', + tokenId: '3', + address: '0xccc', + }, + ], + }, + }, + }, + }, + }; + + const newStorage = await migration77.migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + TokenListController: { + tokensChainsCache: {}, + }, + NftController: { + allNfts: { + '0x333': { + 256: [ + { + name: 'NFT 3', + description: 'Description for NFT 3', + image: 'nft3.jpg', + standard: 'ERC721', + tokenId: '3', + address: '0xccc', + }, + ], + }, + }, + }, + }); + }); + + it('deletes entries in TokenListController.tokensChainsCache that have decimal chain ID keys only if any other chain ID keys are hex', async () => { + const oldStorage = { + meta: { version: 76 }, + data: { + TokenListController: { + tokensChainsCache: { + 16: { + timestamp: 111111, + data: [ + { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + { + address: '0xc00e94cb662c3520282e6f5717214004a7f26888', + symbol: 'COMP', + decimals: 18, + }, + ], + }, + '0x10': { + timestamp: 111111, + data: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + '0xc00e94cb662c3520282e6f5717214004a7f26888': { + address: '0xc00e94cb662c3520282e6f5717214004a7f26888', + symbol: 'COMP', + decimals: 18, + }, + }, + }, + 32: { + timestamp: 222222, + data: [ + { + address: '0x3ee2200efb3400fabb9aacf31297cbdd1d435d47', + symbol: 'ADA', + decimals: 18, + }, + { + address: '0x928e55dab735aa8260af3cedada18b5f70c72f1b', + symbol: 'FRONT', + decimals: 18, + }, + ], + }, + }, + }, + }, + }; + + const newStorage = await migration77.migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + TokenListController: { + tokensChainsCache: { + '0x10': { + timestamp: 111111, + data: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + '0xc00e94cb662c3520282e6f5717214004a7f26888': { + address: '0xc00e94cb662c3520282e6f5717214004a7f26888', + symbol: 'COMP', + decimals: 18, + }, + }, + }, + }, + }, + }); + }); + it('does not delete entries in TokenListController.tokensChainsCache that have decimal chain ID keys if no other chain ID keys are hex', async () => { + const oldStorage = { + meta: { version: 76 }, + data: { + TokenListController: { + tokensChainsCache: { + 16: { + timestamp: 111111, + data: [ + { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + { + address: '0xc00e94cb662c3520282e6f5717214004a7f26888', + symbol: 'COMP', + decimals: 18, + }, + ], + }, + 32: { + timestamp: 222222, + data: [ + { + address: '0x3ee2200efb3400fabb9aacf31297cbdd1d435d47', + symbol: 'ADA', + decimals: 18, + }, + { + address: '0x928e55dab735aa8260af3cedada18b5f70c72f1b', + symbol: 'FRONT', + decimals: 18, + }, + ], + }, + }, + }, + }, + }; + + const newStorage = await migration77.migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + TokenListController: { + tokensChainsCache: { + 16: { + timestamp: 111111, + data: { + '0x514910771af9ca656af840dff83e8264ecf986ca': { + address: '0x514910771af9ca656af840dff83e8264ecf986ca', + symbol: 'LINK', + decimals: 18, + }, + '0xc00e94cb662c3520282e6f5717214004a7f26888': { + address: '0xc00e94cb662c3520282e6f5717214004a7f26888', + symbol: 'COMP', + decimals: 18, + }, + }, + }, + 32: { + timestamp: 222222, + data: { + '0x3ee2200efb3400fabb9aacf31297cbdd1d435d47': { + address: '0x3ee2200efb3400fabb9aacf31297cbdd1d435d47', + symbol: 'ADA', + decimals: 18, + }, + '0x928e55dab735aa8260af3cedada18b5f70c72f1b': { + address: '0x928e55dab735aa8260af3cedada18b5f70c72f1b', + symbol: 'FRONT', + decimals: 18, + }, + }, + }, + }, + }, + }); + }); + it('deletes entries in TokensController.allTokens that have decimal chain IDs only if any other chain ID keys are hex', async () => { + const oldStorage = { + meta: { version: 76 }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + TokensController: { + allTokens: { + 16: { + '0x111': [ + { + address: '0xaaa', + decimals: 1, + symbol: 'TEST1', + }, + ], + }, + '0x10': { + '0x111': [ + { + address: '0xaaa', + decimals: 1, + symbol: 'TEST1', + }, + ], + }, + 32: { + '0x222': [ + { + address: '0xbbb', + decimals: 1, + symbol: 'TEST2', + }, + ], + }, + }, + }, + }, + }; + + const newStorage = await migration77.migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + TokenListController: { + tokensChainsCache: {}, + }, + TokensController: { + allTokens: { + '0x10': { + '0x111': [ + { + address: '0xaaa', + decimals: 1, + symbol: 'TEST1', + }, + ], + }, + }, + }, + }); + }); + + it('does not delete entries in TokensController.allTokens that have decimal chain IDs if no other chain ID keys are hex', async () => { + const oldStorage = { + meta: { version: 76 }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + TokensController: { + allTokens: { + 16: { + '0x111': [ + { + address: '0xaaa', + decimals: 1, + symbol: 'TEST1', + }, + ], + }, + 32: { + '0x222': [ + { + address: '0xbbb', + decimals: 1, + symbol: 'TEST2', + }, + ], + }, + }, + }, + }, + }; + + const newStorage = await migration77.migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + TokenListController: { + tokensChainsCache: {}, + }, + TokensController: { + allTokens: { + 16: { + '0x111': [ + { + address: '0xaaa', + decimals: 1, + symbol: 'TEST1', + }, + ], + }, + 32: { + '0x222': [ + { + address: '0xbbb', + decimals: 1, + symbol: 'TEST2', + }, + ], + }, + }, + }, + }); + }); + + it('deletes entries in TokensController.allIgnoredTokens that have decimal chain IDs only if any other chain ID keys are hex', async () => { + const oldStorage = { + meta: { version: 87 }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + TokensController: { + allIgnoredTokens: { + 16: { + '0x1': { + '0x111': ['0xaaa'], + }, + }, + '0x10': { + '0x1': { + '0x222': ['0xbbb'], + }, + }, + 32: { + '0x2': { + '0x222': ['0xbbb'], + }, + }, + }, + }, + }, + }; + + const newStorage = await migration77.migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + TokenListController: { + tokensChainsCache: {}, + }, + TokensController: { + allIgnoredTokens: { + '0x10': { + '0x1': { + '0x222': ['0xbbb'], + }, + }, + }, + }, + }); + }); + + it('does not delete entries in TokensController.allIgnoredTokens that have decimal chain IDs if no other chain ID keys are hex', async () => { + const oldStorage = { + meta: { version: 87 }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + TokensController: { + allIgnoredTokens: { + 16: { + '0x1': { + '0x111': ['0xaaa'], + }, + }, + 32: { + '0x2': { + '0x222': ['0xbbb'], + }, + }, + }, + }, + }, + }; + + const newStorage = await migration77.migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + TokenListController: { + tokensChainsCache: {}, + }, + TokensController: { + allIgnoredTokens: { + 16: { + '0x1': { + '0x111': ['0xaaa'], + }, + }, + 32: { + '0x2': { + '0x222': ['0xbbb'], + }, + }, + }, + }, + }); + }); + + it('deletes entries in TokensController.allDetectedTokens that have decimal chain IDs only if any other chain ID keys are hex', async () => { + const oldStorage = { + meta: { version: 87 }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + TokensController: { + allDetectedTokens: { + 16: { + '0x1': { + '0x111': ['0xaaa'], + }, + }, + '0x10': { + '0x1': { + '0x222': ['0xbbb'], + }, + }, + 32: { + '0x2': { + '0x222': ['0xbbb'], + }, + }, + }, + }, + }, + }; + + const newStorage = await migration77.migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + TokenListController: { + tokensChainsCache: {}, + }, + TokensController: { + allDetectedTokens: { + '0x10': { + '0x1': { + '0x222': ['0xbbb'], + }, + }, + }, + }, + }); + }); + + it('does not delete entries in TokensController.allDetectedTokens that have decimal chain IDs if no other chain ID keys are hex', async () => { + const oldStorage = { + meta: { version: 87 }, + data: { + TokenListController: { + tokensChainsCache: {}, + }, + TokensController: { + allDetectedTokens: { + 16: { + '0x1': { + '0x111': ['0xaaa'], + }, + }, + 32: { + '0x2': { + '0x222': ['0xbbb'], + }, + }, + }, + }, + }, + }; + + const newStorage = await migration77.migrate(oldStorage); + + expect(newStorage.data).toStrictEqual({ + TokenListController: { + tokensChainsCache: {}, + }, + TokensController: { + allDetectedTokens: { + 16: { + '0x1': { + '0x111': ['0xaaa'], + }, + }, + 32: { + '0x2': { + '0x222': ['0xbbb'], + }, + }, + }, + }, + }); + }); + }); + }); }); From 36b1991b4417ae7292495e884d539a572f31f6c0 Mon Sep 17 00:00:00 2001 From: Mark Stacey Date: Mon, 31 Jul 2023 20:36:14 -0230 Subject: [PATCH 24/25] Fix `prep-deps` CI step (#20312) The CI job `prep-deps` was broken in #20096 for forks and non-PR builds (e.g. the `develop` branch builds). Non-PR builds were broken because of the `set -u` flag, which complained about a PR-specific environment variable being unset. Fork builds were broken because the draft check relied upon secrets (which aren't included in CI runs on forks). The script is now tolerant of missing environment variables, and skips the draft check for forks. --- .circleci/scripts/install-dependencies.sh | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/.circleci/scripts/install-dependencies.sh b/.circleci/scripts/install-dependencies.sh index 78c68c549b9b..35b7a690fa0c 100755 --- a/.circleci/scripts/install-dependencies.sh +++ b/.circleci/scripts/install-dependencies.sh @@ -1,15 +1,17 @@ #!/usr/bin/env bash set -e -set -u set -o pipefail -PR_NUMBER="${CIRCLE_PULL_REQUEST##*/}" -if [ -n "$PR_NUMBER" ] +IS_NON_FORK_DRAFT='false' + +if [[ -n $CIRCLE_PULL_REQUEST ]] && gh auth status then - IS_DRAFT="$(gh pr view --json isDraft --jq '.isDraft' "$PR_NUMBER")" -else - IS_DRAFT='false' + PR_NUMBER="${CIRCLE_PULL_REQUEST##*/}" + if [ -n "$PR_NUMBER" ] + then + IS_NON_FORK_DRAFT="$(gh pr view --json isDraft --jq '.isDraft' "$PR_NUMBER")" + fi fi # Build query to see whether there are any "preview-like" packages in the manifest @@ -29,7 +31,7 @@ else HAS_PREVIEW_BUILDS='false' fi -if [[ $IS_DRAFT == 'true' && $HAS_PREVIEW_BUILDS == 'true' ]] +if [[ $IS_NON_FORK_DRAFT == 'true' && $HAS_PREVIEW_BUILDS == 'true' ]] then # Use GitHub registry on draft PRs, allowing the use of preview builds echo "Installing with preview builds" From 990dc33fc6a9c359ed7b2cb54bee3ceecf1ff90a Mon Sep 17 00:00:00 2001 From: Mark Stacey Date: Mon, 31 Jul 2023 22:26:40 -0230 Subject: [PATCH 25/25] Remove fallback phishing warning configuration (#20327) * Remove fallback phishing warning configuration The package `@metamask/phishing-controller` has been updated from v4 v6. The only breaking changes are a minimum Node.js version bump, and the removal of the fallback phishing configuration. The fallback phishing configuration was resulting in MetaMask being incorrectly flagged as malware, and the stale config was causing problems for sites that had been blocked in the past but have since been unblocked. This should substantially reduce the bundle size as well. * Update LavaMoat policies * Update test state to include example blocked site --------- Co-authored-by: MetaMask Bot --- app/scripts/metamask-controller.test.js | 22 ++++++++++++++++++--- lavamoat/browserify/beta/policy.json | 26 ++----------------------- lavamoat/browserify/desktop/policy.json | 26 ++----------------------- lavamoat/browserify/flask/policy.json | 26 ++----------------------- lavamoat/browserify/main/policy.json | 26 ++----------------------- lavamoat/browserify/mmi/policy.json | 26 ++----------------------- package.json | 2 +- yarn.lock | 16 +++++++-------- 8 files changed, 38 insertions(+), 132 deletions(-) diff --git a/app/scripts/metamask-controller.test.js b/app/scripts/metamask-controller.test.js index b29fce443d78..02d7b1f50ff4 100644 --- a/app/scripts/metamask-controller.test.js +++ b/app/scripts/metamask-controller.test.js @@ -177,6 +177,18 @@ const firstTimeState = { }, }, }, + PhishingController: { + phishingLists: [ + { + allowlist: [], + blocklist: ['test.metamask-phishing.io'], + fuzzylist: [], + tolerance: 0, + version: 0, + name: 'MetaMask', + }, + ], + }, }; const noop = () => undefined; @@ -205,7 +217,7 @@ describe('MetaMaskController', function () { eth_phishing_detect_config: { fuzzylist: [], allowlist: [], - blocklist: ['127.0.0.1'], + blocklist: ['test.metamask-phishing.io'], name: ListNames.MetaMask, }, phishfort_hotlist: { @@ -218,7 +230,11 @@ describe('MetaMaskController', function () { .reply( 200, JSON.stringify([ - { url: '127.0.0.1', targetList: 'blocklist', timestamp: 0 }, + { + url: 'test.metamask-phishing.io', + targetList: 'blocklist', + timestamp: 0, + }, ]), ); @@ -963,7 +979,7 @@ describe('MetaMaskController', function () { it('sets up phishing stream for untrusted communication', async function () { const phishingMessageSender = { - url: 'http://myethereumwalletntw.com', + url: 'http://test.metamask-phishing.io', tab: {}, }; diff --git a/lavamoat/browserify/beta/policy.json b/lavamoat/browserify/beta/policy.json index afb8f0b991df..e1d6a9326196 100644 --- a/lavamoat/browserify/beta/policy.json +++ b/lavamoat/browserify/beta/policy.json @@ -1772,34 +1772,12 @@ "fetch": true }, "packages": { - "@metamask/phishing-controller>@metamask/base-controller": true, - "@metamask/phishing-controller>@metamask/controller-utils": true, + "@metamask/base-controller": true, + "@metamask/controller-utils": true, "@metamask/phishing-warning>eth-phishing-detect": true, "punycode": true } }, - "@metamask/phishing-controller>@metamask/base-controller": { - "packages": { - "immer": true - } - }, - "@metamask/phishing-controller>@metamask/controller-utils": { - "globals": { - "URL": true, - "console.error": true, - "fetch": true, - "setTimeout": true - }, - "packages": { - "@metamask/controller-utils>@spruceid/siwe-parser": true, - "@metamask/utils": true, - "browserify>buffer": true, - "eslint>fast-deep-equal": true, - "eth-ens-namehash": true, - "ethereumjs-util": true, - "ethjs>ethjs-unit": true - } - }, "@metamask/phishing-warning>eth-phishing-detect": { "packages": { "eslint>optionator>fast-levenshtein": true diff --git a/lavamoat/browserify/desktop/policy.json b/lavamoat/browserify/desktop/policy.json index a6805b21fab7..4f3e3f3f4923 100644 --- a/lavamoat/browserify/desktop/policy.json +++ b/lavamoat/browserify/desktop/policy.json @@ -1930,34 +1930,12 @@ "fetch": true }, "packages": { - "@metamask/phishing-controller>@metamask/base-controller": true, - "@metamask/phishing-controller>@metamask/controller-utils": true, + "@metamask/base-controller": true, + "@metamask/controller-utils": true, "@metamask/phishing-warning>eth-phishing-detect": true, "punycode": true } }, - "@metamask/phishing-controller>@metamask/base-controller": { - "packages": { - "immer": true - } - }, - "@metamask/phishing-controller>@metamask/controller-utils": { - "globals": { - "URL": true, - "console.error": true, - "fetch": true, - "setTimeout": true - }, - "packages": { - "@metamask/controller-utils>@spruceid/siwe-parser": true, - "@metamask/utils": true, - "browserify>buffer": true, - "eslint>fast-deep-equal": true, - "eth-ens-namehash": true, - "ethereumjs-util": true, - "ethjs>ethjs-unit": true - } - }, "@metamask/phishing-warning>eth-phishing-detect": { "packages": { "eslint>optionator>fast-levenshtein": true diff --git a/lavamoat/browserify/flask/policy.json b/lavamoat/browserify/flask/policy.json index a6805b21fab7..4f3e3f3f4923 100644 --- a/lavamoat/browserify/flask/policy.json +++ b/lavamoat/browserify/flask/policy.json @@ -1930,34 +1930,12 @@ "fetch": true }, "packages": { - "@metamask/phishing-controller>@metamask/base-controller": true, - "@metamask/phishing-controller>@metamask/controller-utils": true, + "@metamask/base-controller": true, + "@metamask/controller-utils": true, "@metamask/phishing-warning>eth-phishing-detect": true, "punycode": true } }, - "@metamask/phishing-controller>@metamask/base-controller": { - "packages": { - "immer": true - } - }, - "@metamask/phishing-controller>@metamask/controller-utils": { - "globals": { - "URL": true, - "console.error": true, - "fetch": true, - "setTimeout": true - }, - "packages": { - "@metamask/controller-utils>@spruceid/siwe-parser": true, - "@metamask/utils": true, - "browserify>buffer": true, - "eslint>fast-deep-equal": true, - "eth-ens-namehash": true, - "ethereumjs-util": true, - "ethjs>ethjs-unit": true - } - }, "@metamask/phishing-warning>eth-phishing-detect": { "packages": { "eslint>optionator>fast-levenshtein": true diff --git a/lavamoat/browserify/main/policy.json b/lavamoat/browserify/main/policy.json index afb8f0b991df..e1d6a9326196 100644 --- a/lavamoat/browserify/main/policy.json +++ b/lavamoat/browserify/main/policy.json @@ -1772,34 +1772,12 @@ "fetch": true }, "packages": { - "@metamask/phishing-controller>@metamask/base-controller": true, - "@metamask/phishing-controller>@metamask/controller-utils": true, + "@metamask/base-controller": true, + "@metamask/controller-utils": true, "@metamask/phishing-warning>eth-phishing-detect": true, "punycode": true } }, - "@metamask/phishing-controller>@metamask/base-controller": { - "packages": { - "immer": true - } - }, - "@metamask/phishing-controller>@metamask/controller-utils": { - "globals": { - "URL": true, - "console.error": true, - "fetch": true, - "setTimeout": true - }, - "packages": { - "@metamask/controller-utils>@spruceid/siwe-parser": true, - "@metamask/utils": true, - "browserify>buffer": true, - "eslint>fast-deep-equal": true, - "eth-ens-namehash": true, - "ethereumjs-util": true, - "ethjs>ethjs-unit": true - } - }, "@metamask/phishing-warning>eth-phishing-detect": { "packages": { "eslint>optionator>fast-levenshtein": true diff --git a/lavamoat/browserify/mmi/policy.json b/lavamoat/browserify/mmi/policy.json index 63c7e9bb29f3..18882ff84d2f 100644 --- a/lavamoat/browserify/mmi/policy.json +++ b/lavamoat/browserify/mmi/policy.json @@ -2000,34 +2000,12 @@ "fetch": true }, "packages": { - "@metamask/phishing-controller>@metamask/base-controller": true, - "@metamask/phishing-controller>@metamask/controller-utils": true, + "@metamask/base-controller": true, + "@metamask/controller-utils": true, "@metamask/phishing-warning>eth-phishing-detect": true, "punycode": true } }, - "@metamask/phishing-controller>@metamask/base-controller": { - "packages": { - "immer": true - } - }, - "@metamask/phishing-controller>@metamask/controller-utils": { - "globals": { - "URL": true, - "console.error": true, - "fetch": true, - "setTimeout": true - }, - "packages": { - "@metamask/controller-utils>@spruceid/siwe-parser": true, - "@metamask/utils": true, - "browserify>buffer": true, - "eslint>fast-deep-equal": true, - "eth-ens-namehash": true, - "ethereumjs-util": true, - "ethjs>ethjs-unit": true - } - }, "@metamask/phishing-warning>eth-phishing-detect": { "packages": { "eslint>optionator>fast-levenshtein": true diff --git a/package.json b/package.json index 0cc86e215d9c..fcf15f085f53 100644 --- a/package.json +++ b/package.json @@ -256,7 +256,7 @@ "@metamask/notification-controller": "^3.0.0", "@metamask/obs-store": "^8.1.0", "@metamask/permission-controller": "^4.0.0", - "@metamask/phishing-controller": "^4.0.0", + "@metamask/phishing-controller": "^6.0.0", "@metamask/post-message-stream": "^6.0.0", "@metamask/ppom-validator": "^0.1.2", "@metamask/providers": "^11.1.0", diff --git a/yarn.lock b/yarn.lock index 2d7aa26eb1e2..4a6f2c55a280 100644 --- a/yarn.lock +++ b/yarn.lock @@ -3909,7 +3909,7 @@ __metadata: languageName: node linkType: hard -"@metamask/controller-utils@npm:^3.0.0, @metamask/controller-utils@npm:^3.1.0, @metamask/controller-utils@npm:^3.4.0": +"@metamask/controller-utils@npm:^3.0.0, @metamask/controller-utils@npm:^3.4.0": version: 3.4.0 resolution: "@metamask/controller-utils@npm:3.4.0" dependencies: @@ -4494,16 +4494,16 @@ __metadata: languageName: node linkType: hard -"@metamask/phishing-controller@npm:^4.0.0": - version: 4.0.0 - resolution: "@metamask/phishing-controller@npm:4.0.0" +"@metamask/phishing-controller@npm:^6.0.0": + version: 6.0.0 + resolution: "@metamask/phishing-controller@npm:6.0.0" dependencies: - "@metamask/base-controller": ^2.0.0 - "@metamask/controller-utils": ^3.1.0 + "@metamask/base-controller": ^3.2.0 + "@metamask/controller-utils": ^4.3.0 "@types/punycode": ^2.1.0 eth-phishing-detect: ^1.2.0 punycode: ^2.1.1 - checksum: 15de581f7bec21d75531167275c68d7bbeae7fdaad02268749ba0a71c4d3ccb53718d963d6583e90c337407f65b7fcc9a89eb76c6f731802c2668a8425d5df89 + checksum: 13a85865cef1515f6d0ee1cd02da37e5e6b98c493676e3a80195294725b717aa17651a0c24d2e841f790bbd22ae16911cc16bab7846da8266f4ee03007a17f4e languageName: node linkType: hard @@ -24273,7 +24273,7 @@ __metadata: "@metamask/notification-controller": ^3.0.0 "@metamask/obs-store": ^8.1.0 "@metamask/permission-controller": ^4.0.0 - "@metamask/phishing-controller": ^4.0.0 + "@metamask/phishing-controller": ^6.0.0 "@metamask/phishing-warning": ^2.1.0 "@metamask/post-message-stream": ^6.0.0 "@metamask/ppom-validator": ^0.1.2