mirror of
https://github.com/kremalicious/metamask-extension.git
synced 2024-11-22 18:00:18 +01:00
Fix state in confirm transaction (#17838)
This commit is contained in:
parent
1d733194ea
commit
5691a00e20
@ -655,14 +655,16 @@ class FixtureBuilder {
|
||||
chainId: CHAIN_IDS.LOCALHOST,
|
||||
dappSuggestedGasFees: {
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
},
|
||||
history: [
|
||||
{
|
||||
chainId: CHAIN_IDS.LOCALHOST,
|
||||
dappSuggestedGasFees: {
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
},
|
||||
id: 7911313280012623,
|
||||
loadingDefaults: true,
|
||||
@ -673,7 +675,8 @@ class FixtureBuilder {
|
||||
txParams: {
|
||||
from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1',
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
to: '0x2f318c334780961fb129d2a6c30d0763d9a5c970',
|
||||
value: '0x29a2241af62c0000',
|
||||
},
|
||||
@ -698,7 +701,8 @@ class FixtureBuilder {
|
||||
txParams: {
|
||||
from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1',
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
to: '0x2f318c334780961fb129d2a6c30d0763d9a5c970',
|
||||
value: '0x29a2241af62c0000',
|
||||
},
|
||||
@ -708,14 +712,16 @@ class FixtureBuilder {
|
||||
chainId: CHAIN_IDS.LOCALHOST,
|
||||
dappSuggestedGasFees: {
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
},
|
||||
history: [
|
||||
{
|
||||
chainId: CHAIN_IDS.LOCALHOST,
|
||||
dappSuggestedGasFees: {
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
},
|
||||
id: 7911313280012624,
|
||||
loadingDefaults: true,
|
||||
@ -726,7 +732,8 @@ class FixtureBuilder {
|
||||
txParams: {
|
||||
from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1',
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
to: '0x2f318c334780961fb129d2a6c30d0763d9a5c970',
|
||||
value: '0x29a2241af62c0000',
|
||||
},
|
||||
@ -751,7 +758,8 @@ class FixtureBuilder {
|
||||
txParams: {
|
||||
from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1',
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
to: '0x2f318c334780961fb129d2a6c30d0763d9a5c970',
|
||||
value: '0x29a2241af62c0000',
|
||||
},
|
||||
@ -761,14 +769,16 @@ class FixtureBuilder {
|
||||
chainId: CHAIN_IDS.LOCALHOST,
|
||||
dappSuggestedGasFees: {
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
},
|
||||
history: [
|
||||
{
|
||||
chainId: CHAIN_IDS.LOCALHOST,
|
||||
dappSuggestedGasFees: {
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
},
|
||||
id: 7911313280012625,
|
||||
loadingDefaults: true,
|
||||
@ -779,7 +789,8 @@ class FixtureBuilder {
|
||||
txParams: {
|
||||
from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1',
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
to: '0x2f318c334780961fb129d2a6c30d0763d9a5c970',
|
||||
value: '0x29a2241af62c0000',
|
||||
},
|
||||
@ -804,7 +815,8 @@ class FixtureBuilder {
|
||||
txParams: {
|
||||
from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1',
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
to: '0x2f318c334780961fb129d2a6c30d0763d9a5c970',
|
||||
value: '0x29a2241af62c0000',
|
||||
},
|
||||
@ -814,14 +826,16 @@ class FixtureBuilder {
|
||||
chainId: CHAIN_IDS.LOCALHOST,
|
||||
dappSuggestedGasFees: {
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
},
|
||||
history: [
|
||||
{
|
||||
chainId: CHAIN_IDS.LOCALHOST,
|
||||
dappSuggestedGasFees: {
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
},
|
||||
id: 7911313280012626,
|
||||
loadingDefaults: true,
|
||||
@ -832,7 +846,8 @@ class FixtureBuilder {
|
||||
txParams: {
|
||||
from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1',
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
to: '0x2f318c334780961fb129d2a6c30d0763d9a5c970',
|
||||
value: '0x29a2241af62c0000',
|
||||
},
|
||||
@ -857,7 +872,8 @@ class FixtureBuilder {
|
||||
txParams: {
|
||||
from: '0x5cfe73b6021e818b776b421b1c4db2474086a7e1',
|
||||
gas: '0x5208',
|
||||
gasPrice: '0x4a817c800',
|
||||
maxFeePerGas: '0x59682f0c',
|
||||
maxPriorityFeePerGas: '0x59682f00',
|
||||
to: '0x2f318c334780961fb129d2a6c30d0763d9a5c970',
|
||||
value: '0x29a2241af62c0000',
|
||||
},
|
||||
|
98
test/e2e/tests/multiple-transactions.spec.js
Normal file
98
test/e2e/tests/multiple-transactions.spec.js
Normal file
@ -0,0 +1,98 @@
|
||||
const { convertToHexValue, withFixtures } = require('../helpers');
|
||||
const FixtureBuilder = require('../fixture-builder');
|
||||
|
||||
describe('Confirm transactions', function () {
|
||||
const ganacheOptions = {
|
||||
hardfork: 'london',
|
||||
accounts: [
|
||||
{
|
||||
secretKey:
|
||||
'0x7C9529A67102755B7E6102D6D950AC5D5863C98713805CEC576B945B15B71EAC',
|
||||
balance: convertToHexValue(25000000000000000000),
|
||||
},
|
||||
],
|
||||
};
|
||||
it('should be able to confirm multiple transactions', async function () {
|
||||
await withFixtures(
|
||||
{
|
||||
fixtures: new FixtureBuilder()
|
||||
.withTransactionControllerMultipleTransactions()
|
||||
.build(),
|
||||
ganacheOptions,
|
||||
title: this.test.title,
|
||||
},
|
||||
async ({ driver }) => {
|
||||
await driver.navigate();
|
||||
await driver.fill('#password', 'correct horse battery staple');
|
||||
await driver.press('#password', driver.Key.ENTER);
|
||||
|
||||
// confirm multiple transactions
|
||||
await driver.waitForSelector({
|
||||
text: 'Reject 4 transactions',
|
||||
tag: 'a',
|
||||
});
|
||||
await driver.clickElement({ text: 'Confirm', tag: 'button' });
|
||||
await driver.waitForSelector({
|
||||
text: 'Reject 3 transactions',
|
||||
tag: 'a',
|
||||
});
|
||||
await driver.clickElement({ text: 'Confirm', tag: 'button' });
|
||||
await driver.waitForSelector({
|
||||
text: 'Reject 2 transactions',
|
||||
tag: 'a',
|
||||
});
|
||||
await driver.clickElement({ text: 'Confirm', tag: 'button' });
|
||||
|
||||
await driver.waitForElementNotPresent('.loading-overlay__spinner');
|
||||
await driver.clickElement({ text: 'Confirm', tag: 'button' });
|
||||
|
||||
await driver.clickElement('[data-testid="home__activity-tab"]');
|
||||
await driver.wait(async () => {
|
||||
const confirmedTxes = await driver.findElements(
|
||||
'.transaction-list__completed-transactions .transaction-list-item',
|
||||
);
|
||||
return confirmedTxes.length === 4;
|
||||
}, 10000);
|
||||
},
|
||||
);
|
||||
});
|
||||
|
||||
it('should be able to reject multiple transactions', async function () {
|
||||
await withFixtures(
|
||||
{
|
||||
fixtures: new FixtureBuilder()
|
||||
.withTransactionControllerMultipleTransactions()
|
||||
.build(),
|
||||
ganacheOptions,
|
||||
title: this.test.title,
|
||||
},
|
||||
async ({ driver }) => {
|
||||
await driver.navigate();
|
||||
await driver.fill('#password', 'correct horse battery staple');
|
||||
await driver.press('#password', driver.Key.ENTER);
|
||||
|
||||
// confirm multiple transactions
|
||||
await driver.waitForSelector({
|
||||
text: 'Reject 4 transactions',
|
||||
tag: 'a',
|
||||
});
|
||||
await driver.clickElement({ text: 'Reject', tag: 'button' });
|
||||
await driver.waitForSelector({
|
||||
text: 'Reject 3 transactions',
|
||||
tag: 'a',
|
||||
});
|
||||
await driver.clickElement({ text: 'Reject', tag: 'button' });
|
||||
await driver.waitForSelector({
|
||||
text: 'Reject 2 transactions',
|
||||
tag: 'a',
|
||||
});
|
||||
await driver.clickElement({ text: 'Reject', tag: 'button' });
|
||||
|
||||
await driver.waitForElementNotPresent('.loading-overlay__spinner');
|
||||
await driver.clickElement({ text: 'Reject', tag: 'button' });
|
||||
|
||||
await driver.waitForSelector('[data-testid="home__activity-tab"]');
|
||||
},
|
||||
);
|
||||
});
|
||||
});
|
@ -4,6 +4,7 @@ const FixtureBuilder = require('../fixture-builder');
|
||||
|
||||
describe('Navigate transactions', function () {
|
||||
const ganacheOptions = {
|
||||
hardfork: 'london',
|
||||
accounts: [
|
||||
{
|
||||
secretKey:
|
||||
|
@ -776,14 +776,12 @@ export default class ConfirmTransactionBase extends Component {
|
||||
cancelTransaction,
|
||||
history,
|
||||
mostRecentOverviewPage,
|
||||
clearConfirmTransaction,
|
||||
updateCustomNonce,
|
||||
} = this.props;
|
||||
|
||||
this._removeBeforeUnload();
|
||||
updateCustomNonce('');
|
||||
cancelTransaction(txData).then(() => {
|
||||
clearConfirmTransaction();
|
||||
history.push(mostRecentOverviewPage);
|
||||
});
|
||||
}
|
||||
@ -791,7 +789,6 @@ export default class ConfirmTransactionBase extends Component {
|
||||
handleSubmit() {
|
||||
const {
|
||||
sendTransaction,
|
||||
clearConfirmTransaction,
|
||||
txData,
|
||||
history,
|
||||
mostRecentOverviewPage,
|
||||
@ -865,7 +862,6 @@ export default class ConfirmTransactionBase extends Component {
|
||||
|
||||
sendTransaction(txData)
|
||||
.then(() => {
|
||||
clearConfirmTransaction();
|
||||
if (!this._isMounted) {
|
||||
return;
|
||||
}
|
||||
@ -995,6 +991,7 @@ export default class ConfirmTransactionBase extends Component {
|
||||
componentWillUnmount() {
|
||||
this._beforeUnloadForGasPolling();
|
||||
this._removeBeforeUnload();
|
||||
this.props.clearConfirmTransaction();
|
||||
}
|
||||
|
||||
supportsEIP1559 =
|
||||
|
@ -1,4 +1,4 @@
|
||||
import React, { useCallback, useEffect, useMemo, useState } from 'react';
|
||||
import React, { useCallback, useEffect, useState } from 'react';
|
||||
import { useDispatch, useSelector } from 'react-redux';
|
||||
import { Switch, Route, useHistory, useParams } from 'react-router-dom';
|
||||
|
||||
@ -61,8 +61,7 @@ const ConfirmTransaction = () => {
|
||||
const unconfirmedMessages = useSelector(unconfirmedTransactionsHashSelector);
|
||||
|
||||
const totalUnapproved = unconfirmedTxs.length || 0;
|
||||
|
||||
const transaction = useMemo(() => {
|
||||
const getTransaction = useCallback(() => {
|
||||
return totalUnapproved
|
||||
? unapprovedTxs[paramsTransactionId] ||
|
||||
unconfirmedMessages[paramsTransactionId] ||
|
||||
@ -75,10 +74,27 @@ const ConfirmTransaction = () => {
|
||||
unconfirmedMessages,
|
||||
unconfirmedTxs,
|
||||
]);
|
||||
const [transaction, setTransaction] = useState(getTransaction);
|
||||
|
||||
useEffect(() => {
|
||||
const tx = getTransaction();
|
||||
setTransaction(tx);
|
||||
if (tx?.id) {
|
||||
dispatch(setTransactionToConfirm(tx.id));
|
||||
}
|
||||
}, [
|
||||
dispatch,
|
||||
getTransaction,
|
||||
paramsTransactionId,
|
||||
totalUnapproved,
|
||||
unapprovedTxs,
|
||||
unconfirmedMessages,
|
||||
unconfirmedTxs,
|
||||
]);
|
||||
|
||||
const { id, type } = transaction;
|
||||
const transactionId = id && String(id);
|
||||
const isValidERC20TokenMethod = isTokenMethodAction(type);
|
||||
const isValidTokenMethod = isTokenMethodAction(type);
|
||||
const isValidTransactionId =
|
||||
transactionId &&
|
||||
(!paramsTransactionId || paramsTransactionId === transactionId);
|
||||
@ -152,7 +168,8 @@ const ConfirmTransaction = () => {
|
||||
} else if (
|
||||
prevTransactionId &&
|
||||
transactionId &&
|
||||
prevTransactionId !== transactionId
|
||||
prevTransactionId !== transactionId &&
|
||||
paramsTransactionId !== transactionId
|
||||
) {
|
||||
history.replace(mostRecentOverviewPage);
|
||||
}
|
||||
@ -168,7 +185,7 @@ const ConfirmTransaction = () => {
|
||||
transactionId,
|
||||
]);
|
||||
|
||||
if (isValidERC20TokenMethod && isValidTransactionId) {
|
||||
if (isValidTokenMethod && isValidTransactionId) {
|
||||
return <ConfirmTokenTransactionSwitch transaction={transaction} />;
|
||||
}
|
||||
// Show routes when state.confirmTransaction has been set and when either the ID in the params
|
||||
|
Loading…
Reference in New Issue
Block a user