From f71700662bfc2cb95e358cd59a05f3664275e250 Mon Sep 17 00:00:00 2001 From: iamacook Date: Wed, 5 Jul 2023 11:46:14 +0200 Subject: [PATCH 1/3] fix: underpriced transactions on Ledger devices --- .../new-safe/create/steps/ReviewStep/index.tsx | 12 ++++++++++-- .../create/steps/StatusStep/useSafeCreation.ts | 13 +++++++++++-- .../BatchExecuteModal/ReviewBatchExecute.tsx | 12 +++++++++--- src/services/tx/tx-sender/dispatch.ts | 8 ++++++-- src/utils/transactions.ts | 16 ++++++++++++++++ 5 files changed, 52 insertions(+), 9 deletions(-) diff --git a/src/components/new-safe/create/steps/ReviewStep/index.tsx b/src/components/new-safe/create/steps/ReviewStep/index.tsx index 54a2bfc71e..be98ef6dd2 100644 --- a/src/components/new-safe/create/steps/ReviewStep/index.tsx +++ b/src/components/new-safe/create/steps/ReviewStep/index.tsx @@ -26,6 +26,7 @@ import { ExecutionMethodSelector, ExecutionMethod } from '@/components/tx/Execut import { useLeastRemainingRelays } from '@/hooks/useRemainingRelays' import classnames from 'classnames' import { hasRemainingRelays } from '@/utils/relaying' +import { BigNumber } from 'ethers' const ReviewStep = ({ data, onSubmit, onBack, setStep }: StepRenderProps) => { const isWrongChain = useIsWrongChain() @@ -56,8 +57,15 @@ const ReviewStep = ({ data, onSubmit, onBack, setStep }: StepRenderProps 0.001' const handleBack = () => { diff --git a/src/components/new-safe/create/steps/StatusStep/useSafeCreation.ts b/src/components/new-safe/create/steps/StatusStep/useSafeCreation.ts index 35a73878a7..0615fcbff8 100644 --- a/src/components/new-safe/create/steps/StatusStep/useSafeCreation.ts +++ b/src/components/new-safe/create/steps/StatusStep/useSafeCreation.ts @@ -20,6 +20,8 @@ import { useAppDispatch } from '@/store' import { closeByGroupKey } from '@/store/notificationsSlice' import { CREATE_SAFE_EVENTS, trackEvent } from '@/services/analytics' import { waitForCreateSafeTx } from '@/services/tx/txMonitor' +import useGasPrice from '@/hooks/useGasPrice' +import { shouldSetGasPrice } from '@/utils/transactions' export enum SafeCreationStatus { AWAITING, @@ -48,6 +50,7 @@ export const useSafeCreation = ( const provider = useWeb3() const web3ReadOnly = useWeb3ReadOnly() const chain = useCurrentChain() + const { maxFeePerGas } = useGasPrice() const createSafeCallback = useCallback( async (txHash: string, tx: PendingSafeTx) => { @@ -59,7 +62,7 @@ export const useSafeCreation = ( ) const handleCreateSafe = useCallback(async () => { - if (!pendingSafe || !provider || !chain || !wallet || isCreating) return + if (!pendingSafe || !provider || !chain || !wallet || isCreating || !maxFeePerGas) return setIsCreating(true) dispatch(closeByGroupKey({ groupKey: SAFE_CREATION_ERROR_KEY })) @@ -87,7 +90,12 @@ export const useSafeCreation = ( chain.chainId, ) - await createNewSafe(provider, safeParams) + await createNewSafe(provider, { + ...safeParams, + options: { + gasPrice: shouldSetGasPrice(chain, wallet) ? maxFeePerGas.toString() : undefined, + }, + }) setStatus(SafeCreationStatus.SUCCESS) } } catch (err) { @@ -107,6 +115,7 @@ export const useSafeCreation = ( createSafeCallback, dispatch, isCreating, + maxFeePerGas, pendingSafe, provider, setPendingSafe, diff --git a/src/components/tx/modals/BatchExecuteModal/ReviewBatchExecute.tsx b/src/components/tx/modals/BatchExecuteModal/ReviewBatchExecute.tsx index b38e0603d9..5f86a2aed0 100644 --- a/src/components/tx/modals/BatchExecuteModal/ReviewBatchExecute.tsx +++ b/src/components/tx/modals/BatchExecuteModal/ReviewBatchExecute.tsx @@ -12,7 +12,7 @@ import { Errors, logError } from '@/services/exceptions' import ErrorMessage from '@/components/tx/ErrorMessage' import type { BatchExecuteData } from '@/components/tx/modals/BatchExecuteModal/index' import DecodedTxs from '@/components/tx/modals/BatchExecuteModal/DecodedTxs' -import { getMultiSendTxs, getTxsWithDetails } from '@/utils/transactions' +import { getMultiSendTxs, getTxsWithDetails, shouldSetGasPrice } from '@/utils/transactions' import { TxSimulation } from '@/components/tx/TxSimulation' import { useRelaysBySafe } from '@/hooks/useRemainingRelays' import { ExecutionMethod, ExecutionMethodSelector } from '../../ExecutionMethodSelector' @@ -21,6 +21,8 @@ import useOnboard from '@/hooks/wallets/useOnboard' import { WrongChainWarning } from '@/components/tx/WrongChainWarning' import { useWeb3 } from '@/hooks/wallets/web3' import { hasRemainingRelays } from '@/utils/relaying' +import useGasPrice from '@/hooks/useGasPrice' +import useWallet from '@/hooks/wallets/useWallet' const ReviewBatchExecute = ({ data, onSubmit }: { data: BatchExecuteData; onSubmit: (data: null) => void }) => { const [isSubmittable, setIsSubmittable] = useState(true) @@ -29,6 +31,8 @@ const ReviewBatchExecute = ({ data, onSubmit }: { data: BatchExecuteData; onSubm const chain = useCurrentChain() const { safe } = useSafeInfo() const [relays] = useRelaysBySafe() + const wallet = useWallet() + const { maxFeePerGas } = useGasPrice() // Chain has relaying feature and available relays const canRelay = hasRemainingRelays(relays) @@ -58,7 +62,8 @@ const ReviewBatchExecute = ({ data, onSubmit }: { data: BatchExecuteData; onSubm }, [txsWithDetails, multiSendTxs]) const onExecute = async () => { - if (!onboard || !multiSendTxData || !multiSendContract || !txsWithDetails) return + if (!onboard || !multiSendTxData || !multiSendContract || !txsWithDetails || !chain || !wallet || !maxFeePerGas) + return await dispatchBatchExecution( txsWithDetails, @@ -67,6 +72,7 @@ const ReviewBatchExecute = ({ data, onSubmit }: { data: BatchExecuteData; onSubm onboard, safe.chainId, safe.address.value, + shouldSetGasPrice(chain, wallet) ? maxFeePerGas : undefined, ) onSubmit(null) } @@ -100,7 +106,7 @@ const ReviewBatchExecute = ({ data, onSubmit }: { data: BatchExecuteData; onSubm } } - const submitDisabled = loading || !isSubmittable + const submitDisabled = loading || !isSubmittable || !maxFeePerGas return (
diff --git a/src/services/tx/tx-sender/dispatch.ts b/src/services/tx/tx-sender/dispatch.ts index ae390acf7c..3d1003c12a 100644 --- a/src/services/tx/tx-sender/dispatch.ts +++ b/src/services/tx/tx-sender/dispatch.ts @@ -5,7 +5,7 @@ import { didReprice, didRevert } from '@/utils/ethers-utils' import type MultiSendCallOnlyEthersContract from '@safe-global/safe-ethers-lib/dist/src/contracts/MultiSendCallOnly/MultiSendCallOnlyEthersContract' import type { SpendingLimitTxParams } from '@/components/tx/modals/TokenTransferModal/ReviewSpendingLimitTx' import { getSpendingLimitContract } from '@/services/contracts/spendingLimitContracts' -import type { ContractTransaction } from 'ethers' +import type { BigNumber, ContractTransaction } from 'ethers' import type { RequestId } from '@safe-global/safe-apps-sdk' import proposeTx from '../proposeTransaction' import { txDispatch, TxEvent } from '../txEvents' @@ -174,6 +174,7 @@ export const dispatchBatchExecution = async ( onboard: OnboardAPI, chainId: SafeInfo['chainId'], safeAddress: string, + gasPrice?: BigNumber, ) => { const groupKey = multiSendTxData @@ -183,7 +184,10 @@ export const dispatchBatchExecution = async ( const wallet = await assertWalletChain(onboard, chainId) const provider = createWeb3(wallet.provider) - result = await multiSendContract.contract.connect(provider.getSigner()).multiSend(multiSendTxData) + result = await multiSendContract.contract + .connect(provider.getSigner()) + .multiSend(multiSendTxData, { gasPrice: gasPrice?.toString() }) + txs.forEach(({ txId }) => { txDispatch(TxEvent.EXECUTING, { txId, groupKey }) }) diff --git a/src/utils/transactions.ts b/src/utils/transactions.ts index f5dd81f4a2..0bba799811 100644 --- a/src/utils/transactions.ts +++ b/src/utils/transactions.ts @@ -30,6 +30,9 @@ import { Multi_send__factory } from '@/types/contracts' import { ethers } from 'ethers' import { type BaseTransaction } from '@safe-global/safe-apps-sdk' import { id } from 'ethers/lib/utils' +import chains from '@/config/chains' +import { WALLET_KEYS } from '@/hooks/wallets/consts' +import type { ConnectedWallet } from '@/services/onboard' export const makeTxFromDetails = (txDetails: TransactionDetails): Transaction => { const getMissingSigners = ({ @@ -266,3 +269,16 @@ export const decodeMultiSendTxs = (encodedMultiSendData: string): BaseTransactio return txs } + +/** + * Transactions are underpriced on Ledger if the gasPrice is not set + */ +export const shouldSetGasPrice = (chain: ChainInfo, wallet: ConnectedWallet): boolean => { + const POLYGON_CHAIN_ID = chains.matic + + const isEIP1559 = hasFeature(chain, FEATURES.EIP1559) + const isPolygon = chains[chain.shortName] === POLYGON_CHAIN_ID + const isLedger = wallet.label.toUpperCase() === WALLET_KEYS.LEDGER + + return !isEIP1559 && isLedger && isPolygon +} From 8bc88488c4d226a835782e5b13941b4aa0f7fb4c Mon Sep 17 00:00:00 2001 From: iamacook Date: Wed, 5 Jul 2023 16:05:48 +0200 Subject: [PATCH 2/3] refactor: use loading flag from `useGasPrice` --- .../create/steps/ReviewStep/index.tsx | 8 +- .../__tests__/useSafeCreation.test.ts | 79 ++++++++++++++++++- .../steps/StatusStep/useSafeCreation.ts | 24 ++++-- .../tx/AdvancedParams/useAdvancedParams.ts | 8 +- .../BatchExecuteModal/ReviewBatchExecute.tsx | 25 ++++-- src/hooks/__tests__/useGasPrice.test.ts | 50 +++++++++--- src/hooks/useGasPrice.ts | 51 ++++++------ src/services/tx/tx-sender/dispatch.ts | 8 +- src/utils/transactions.ts | 16 ---- 9 files changed, 191 insertions(+), 78 deletions(-) diff --git a/src/components/new-safe/create/steps/ReviewStep/index.tsx b/src/components/new-safe/create/steps/ReviewStep/index.tsx index be98ef6dd2..c5ebf035f4 100644 --- a/src/components/new-safe/create/steps/ReviewStep/index.tsx +++ b/src/components/new-safe/create/steps/ReviewStep/index.tsx @@ -34,7 +34,7 @@ const ReviewStep = ({ data, onSubmit, onBack, setStep }: StepRenderProps Date.now(), []) const [_, setPendingSafe] = useLocalStorage(SAFE_PENDING_CREATION_STORAGE_KEY) const [executionMethod, setExecutionMethod] = useState(ExecutionMethod.RELAY) @@ -56,12 +56,16 @@ const ReviewStep = ({ data, onSubmit, onBack, setStep }: StepRenderProps { const mockChain = { chainId: '4', + features: [], } as unknown as ChainInfo jest.spyOn(web3, 'useWeb3').mockImplementation(() => mockProvider) @@ -56,15 +59,89 @@ describe('useSafeCreation', () => { jest .spyOn(contracts, 'getReadOnlyFallbackHandlerContract') .mockReturnValue({ getAddress: () => hexZeroPad('0x123', 20) } as CompatibilityFallbackHandlerEthersContract) + jest + .spyOn(gasPrice, 'default') + .mockReturnValue([{ maxFeePerGas: BigNumber.from(123), maxPriorityFeePerGas: undefined }, undefined, false]) + }) + + it('should create a safe with gas params if there is no txHash and status is AWAITING', async () => { + const createSafeSpy = jest.spyOn(logic, 'createNewSafe').mockReturnValue(Promise.resolve({} as Safe)) + + renderHook(() => useSafeCreation(mockPendingSafe, mockSetPendingSafe, mockStatus, mockSetStatus, false)) + + await waitFor(() => { + expect(createSafeSpy).toHaveBeenCalled() + + const { gasPrice, maxFeePerGas, maxPriorityFeePerGas } = createSafeSpy.mock.calls[0][1].options || {} + + expect(gasPrice).toBe('123') + + expect(maxFeePerGas).toBeUndefined() + expect(maxPriorityFeePerGas).toBeUndefined() + }) + }) + + it('should create a safe with EIP-1559 gas params if there is no txHash and status is AWAITING', async () => { + jest + .spyOn(gasPrice, 'default') + .mockReturnValue([ + { maxFeePerGas: BigNumber.from(123), maxPriorityFeePerGas: BigNumber.from(456) }, + undefined, + false, + ]) + + jest.spyOn(chain, 'useCurrentChain').mockImplementation( + () => + ({ + chainId: '4', + features: [FEATURES.EIP1559], + } as unknown as ChainInfo), + ) + const createSafeSpy = jest.spyOn(logic, 'createNewSafe').mockReturnValue(Promise.resolve({} as Safe)) + + renderHook(() => useSafeCreation(mockPendingSafe, mockSetPendingSafe, mockStatus, mockSetStatus, false)) + + await waitFor(() => { + expect(createSafeSpy).toHaveBeenCalled() + + const { gasPrice, maxFeePerGas, maxPriorityFeePerGas } = createSafeSpy.mock.calls[0][1].options || {} + + expect(maxFeePerGas).toBe('123') + expect(maxPriorityFeePerGas).toBe('456') + + expect(gasPrice).toBeUndefined() + }) }) - it('should create a safe if there is no txHash and status is AWAITING', async () => { + it('should create a safe with no gas params if the gas estimation threw, there is no txHash and status is AWAITING', async () => { + jest + .spyOn(gasPrice, 'default') + .mockReturnValue([{ maxFeePerGas: undefined, maxPriorityFeePerGas: undefined }, undefined, false]) + const createSafeSpy = jest.spyOn(logic, 'createNewSafe').mockReturnValue(Promise.resolve({} as Safe)) renderHook(() => useSafeCreation(mockPendingSafe, mockSetPendingSafe, mockStatus, mockSetStatus, false)) await waitFor(() => { expect(createSafeSpy).toHaveBeenCalled() + + const { gasPrice, maxFeePerGas, maxPriorityFeePerGas } = createSafeSpy.mock.calls[0][1].options || {} + + expect(gasPrice).toBeUndefined() + expect(maxFeePerGas).toBeUndefined() + expect(maxPriorityFeePerGas).toBeUndefined() + }) + }) + + it('should not create a safe if there is no txHash, status is AWAITING but gas is loading', async () => { + jest.spyOn(gasPrice, 'default').mockReturnValue([undefined, undefined, true]) + + const createSafeSpy = jest.spyOn(logic, 'createNewSafe').mockReturnValue(Promise.resolve({} as Safe)) + + renderHook(() => useSafeCreation(mockPendingSafe, mockSetPendingSafe, mockStatus, mockSetStatus, false)) + + await waitFor(() => { + expect(createSafeSpy).not.toHaveBeenCalled() }) }) diff --git a/src/components/new-safe/create/steps/StatusStep/useSafeCreation.ts b/src/components/new-safe/create/steps/StatusStep/useSafeCreation.ts index 0615fcbff8..ec4823a5fa 100644 --- a/src/components/new-safe/create/steps/StatusStep/useSafeCreation.ts +++ b/src/components/new-safe/create/steps/StatusStep/useSafeCreation.ts @@ -21,7 +21,9 @@ import { closeByGroupKey } from '@/store/notificationsSlice' import { CREATE_SAFE_EVENTS, trackEvent } from '@/services/analytics' import { waitForCreateSafeTx } from '@/services/tx/txMonitor' import useGasPrice from '@/hooks/useGasPrice' -import { shouldSetGasPrice } from '@/utils/transactions' +import { hasFeature } from '@/utils/chains' +import { FEATURES } from '@safe-global/safe-gateway-typescript-sdk' +import type { DeploySafeProps } from '@safe-global/safe-core-sdk' export enum SafeCreationStatus { AWAITING, @@ -50,7 +52,12 @@ export const useSafeCreation = ( const provider = useWeb3() const web3ReadOnly = useWeb3ReadOnly() const chain = useCurrentChain() - const { maxFeePerGas } = useGasPrice() + const [gasPrice, , gasPriceLoading] = useGasPrice() + + const maxFeePerGas = gasPrice?.maxFeePerGas + const maxPriorityFeePerGas = gasPrice?.maxPriorityFeePerGas + + const isEIP1559 = chain && hasFeature(chain, FEATURES.EIP1559) const createSafeCallback = useCallback( async (txHash: string, tx: PendingSafeTx) => { @@ -62,7 +69,7 @@ export const useSafeCreation = ( ) const handleCreateSafe = useCallback(async () => { - if (!pendingSafe || !provider || !chain || !wallet || isCreating || !maxFeePerGas) return + if (!pendingSafe || !provider || !chain || !wallet || isCreating || gasPriceLoading) return setIsCreating(true) dispatch(closeByGroupKey({ groupKey: SAFE_CREATION_ERROR_KEY })) @@ -90,11 +97,13 @@ export const useSafeCreation = ( chain.chainId, ) + const options: DeploySafeProps['options'] = isEIP1559 + ? { maxFeePerGas: maxFeePerGas?.toString(), maxPriorityFeePerGas: maxPriorityFeePerGas?.toString() } + : { gasPrice: maxFeePerGas?.toString() } + await createNewSafe(provider, { ...safeParams, - options: { - gasPrice: shouldSetGasPrice(chain, wallet) ? maxFeePerGas.toString() : undefined, - }, + options, }) setStatus(SafeCreationStatus.SUCCESS) } @@ -114,8 +123,11 @@ export const useSafeCreation = ( chain, createSafeCallback, dispatch, + gasPriceLoading, isCreating, + isEIP1559, maxFeePerGas, + maxPriorityFeePerGas, pendingSafe, provider, setPendingSafe, diff --git a/src/components/tx/AdvancedParams/useAdvancedParams.ts b/src/components/tx/AdvancedParams/useAdvancedParams.ts index a03773b7b5..1d876384b4 100644 --- a/src/components/tx/AdvancedParams/useAdvancedParams.ts +++ b/src/components/tx/AdvancedParams/useAdvancedParams.ts @@ -9,7 +9,7 @@ export const useAdvancedParams = ({ safeTxGas, }: AdvancedParameters): [AdvancedParameters, (params: AdvancedParameters) => void] => { const [manualParams, setManualParams] = useState() - const { maxFeePerGas, maxPriorityFeePerGas } = useGasPrice() + const [gasPrice] = useGasPrice() const userNonce = useUserNonce() const advancedParams: AdvancedParameters = useMemo( @@ -17,11 +17,11 @@ export const useAdvancedParams = ({ nonce: manualParams?.nonce ?? nonce, userNonce: manualParams?.userNonce ?? userNonce, gasLimit: manualParams?.gasLimit ?? gasLimit, - maxFeePerGas: manualParams?.maxFeePerGas ?? maxFeePerGas, - maxPriorityFeePerGas: manualParams?.maxPriorityFeePerGas ?? maxPriorityFeePerGas, + maxFeePerGas: manualParams?.maxFeePerGas ?? gasPrice?.maxFeePerGas, + maxPriorityFeePerGas: manualParams?.maxPriorityFeePerGas ?? gasPrice?.maxPriorityFeePerGas, safeTxGas: manualParams?.safeTxGas ?? safeTxGas, }), - [manualParams, nonce, userNonce, gasLimit, maxFeePerGas, maxPriorityFeePerGas, safeTxGas], + [manualParams, nonce, userNonce, gasLimit, gasPrice?.maxFeePerGas, gasPrice?.maxPriorityFeePerGas, safeTxGas], ) return [advancedParams, setManualParams] diff --git a/src/components/tx/modals/BatchExecuteModal/ReviewBatchExecute.tsx b/src/components/tx/modals/BatchExecuteModal/ReviewBatchExecute.tsx index 5f86a2aed0..8c505df00c 100644 --- a/src/components/tx/modals/BatchExecuteModal/ReviewBatchExecute.tsx +++ b/src/components/tx/modals/BatchExecuteModal/ReviewBatchExecute.tsx @@ -1,4 +1,5 @@ import useAsync from '@/hooks/useAsync' +import { FEATURES } from '@safe-global/safe-gateway-typescript-sdk' import type { TransactionDetails } from '@safe-global/safe-gateway-typescript-sdk' import { getMultiSendCallOnlyContract } from '@/services/contracts/safeContracts' import { useCurrentChain } from '@/hooks/useChains' @@ -12,7 +13,7 @@ import { Errors, logError } from '@/services/exceptions' import ErrorMessage from '@/components/tx/ErrorMessage' import type { BatchExecuteData } from '@/components/tx/modals/BatchExecuteModal/index' import DecodedTxs from '@/components/tx/modals/BatchExecuteModal/DecodedTxs' -import { getMultiSendTxs, getTxsWithDetails, shouldSetGasPrice } from '@/utils/transactions' +import { getMultiSendTxs, getTxsWithDetails } from '@/utils/transactions' import { TxSimulation } from '@/components/tx/TxSimulation' import { useRelaysBySafe } from '@/hooks/useRemainingRelays' import { ExecutionMethod, ExecutionMethodSelector } from '../../ExecutionMethodSelector' @@ -22,7 +23,8 @@ import { WrongChainWarning } from '@/components/tx/WrongChainWarning' import { useWeb3 } from '@/hooks/wallets/web3' import { hasRemainingRelays } from '@/utils/relaying' import useGasPrice from '@/hooks/useGasPrice' -import useWallet from '@/hooks/wallets/useWallet' +import { hasFeature } from '@/utils/chains' +import type { PayableOverrides } from 'ethers' const ReviewBatchExecute = ({ data, onSubmit }: { data: BatchExecuteData; onSubmit: (data: null) => void }) => { const [isSubmittable, setIsSubmittable] = useState(true) @@ -31,8 +33,12 @@ const ReviewBatchExecute = ({ data, onSubmit }: { data: BatchExecuteData; onSubm const chain = useCurrentChain() const { safe } = useSafeInfo() const [relays] = useRelaysBySafe() - const wallet = useWallet() - const { maxFeePerGas } = useGasPrice() + const [gasPrice, , gasPriceLoading] = useGasPrice() + + const maxFeePerGas = gasPrice?.maxFeePerGas + const maxPriorityFeePerGas = gasPrice?.maxPriorityFeePerGas + + const isEIP1559 = chain && hasFeature(chain, FEATURES.EIP1559) // Chain has relaying feature and available relays const canRelay = hasRemainingRelays(relays) @@ -62,8 +68,11 @@ const ReviewBatchExecute = ({ data, onSubmit }: { data: BatchExecuteData; onSubm }, [txsWithDetails, multiSendTxs]) const onExecute = async () => { - if (!onboard || !multiSendTxData || !multiSendContract || !txsWithDetails || !chain || !wallet || !maxFeePerGas) - return + if (!onboard || !multiSendTxData || !multiSendContract || !txsWithDetails || gasPriceLoading) return + + const overrides: PayableOverrides = isEIP1559 + ? { maxFeePerGas: maxFeePerGas?.toString(), maxPriorityFeePerGas: maxPriorityFeePerGas?.toString() } + : { gasPrice: maxFeePerGas?.toString() } await dispatchBatchExecution( txsWithDetails, @@ -72,7 +81,7 @@ const ReviewBatchExecute = ({ data, onSubmit }: { data: BatchExecuteData; onSubm onboard, safe.chainId, safe.address.value, - shouldSetGasPrice(chain, wallet) ? maxFeePerGas : undefined, + overrides, ) onSubmit(null) } @@ -106,7 +115,7 @@ const ReviewBatchExecute = ({ data, onSubmit }: { data: BatchExecuteData; onSubm } } - const submitDisabled = loading || !isSubmittable || !maxFeePerGas + const submitDisabled = loading || !isSubmittable || gasPriceLoading return (
diff --git a/src/hooks/__tests__/useGasPrice.test.ts b/src/hooks/__tests__/useGasPrice.test.ts index 987413397c..d3a3d5dd24 100644 --- a/src/hooks/__tests__/useGasPrice.test.ts +++ b/src/hooks/__tests__/useGasPrice.test.ts @@ -74,6 +74,9 @@ describe('useGasPrice', () => { // render the hook const { result } = renderHook(() => useGasPrice()) + // assert the hook is loading + expect(result.current[2]).toBe(true) + // wait for the hook to fetch the gas price await act(async () => { await Promise.resolve() @@ -81,11 +84,14 @@ describe('useGasPrice', () => { expect(fetch).toHaveBeenCalledWith('https://api.etherscan.io/api?module=gastracker&action=gasoracle') + // assert the hook is not loading + expect(result.current[2]).toBe(false) + // assert the gas price is correct - expect(result.current.maxFeePerGas?.toString()).toBe('47000000000') + expect(result.current[0]?.maxFeePerGas?.toString()).toBe('47000000000') // assert the priority fee is correct - expect(result.current.maxPriorityFeePerGas?.toString()).toEqual('4975') + expect(result.current[0]?.maxPriorityFeePerGas?.toString()).toEqual('4975') }) it('should return the fetched gas price from the second oracle if the first one fails', async () => { @@ -110,6 +116,9 @@ describe('useGasPrice', () => { // render the hook const { result } = renderHook(() => useGasPrice()) + // assert the hook is loading + expect(result.current[2]).toBe(true) + // wait for the hook to fetch the gas price await act(async () => { await Promise.resolve() @@ -118,11 +127,14 @@ describe('useGasPrice', () => { expect(fetch).toHaveBeenCalledWith('https://api.etherscan.io/api?module=gastracker&action=gasoracle') expect(fetch).toHaveBeenCalledWith('https://ethgasstation.info/json/ethgasAPI.json') + // assert the hook is not loading + expect(result.current[2]).toBe(false) + // assert the gas price is correct - expect(result.current.maxFeePerGas?.toString()).toBe('60000000000') + expect(result.current[0]?.maxFeePerGas?.toString()).toBe('60000000000') // assert the priority fee is correct - expect(result.current.maxPriorityFeePerGas?.toString()).toEqual('4975') + expect(result.current[0]?.maxPriorityFeePerGas?.toString()).toEqual('4975') }) it('should fallback to a fixed gas price if the oracles fail', async () => { @@ -137,6 +149,9 @@ describe('useGasPrice', () => { // render the hook const { result } = renderHook(() => useGasPrice()) + // assert the hook is loading + expect(result.current[2]).toBe(true) + // wait for the hook to fetch the gas price await act(async () => { await Promise.resolve() @@ -145,11 +160,14 @@ describe('useGasPrice', () => { expect(fetch).toHaveBeenCalledWith('https://api.etherscan.io/api?module=gastracker&action=gasoracle') expect(fetch).toHaveBeenCalledWith('https://ethgasstation.info/json/ethgasAPI.json') + // assert the hook is not loading + expect(result.current[2]).toBe(false) + // assert the gas price is correct - expect(result.current.maxFeePerGas?.toString()).toBe('24000000000') + expect(result.current[0]?.maxFeePerGas?.toString()).toBe('24000000000') // assert the priority fee is correct - expect(result.current.maxPriorityFeePerGas?.toString()).toEqual('4975') + expect(result.current[0]?.maxPriorityFeePerGas?.toString()).toEqual('4975') }) it('should keep the previous gas price if the hook re-renders', async () => { @@ -185,25 +203,37 @@ describe('useGasPrice', () => { // render the hook const { result } = renderHook(() => useGasPrice()) - expect(result.current.maxFeePerGas).toBe(undefined) + // assert the hook is loading + expect(result.current[2]).toBe(true) + + expect(result.current[0]?.maxFeePerGas).toBe(undefined) // wait for the hook to fetch the gas price await act(async () => { await Promise.resolve() }) - expect(result.current.maxFeePerGas?.toString()).toBe('21000000000') + // assert the hook is not loading + expect(result.current[2]).toBe(false) + + expect(result.current[0]?.maxFeePerGas?.toString()).toBe('21000000000') // render the hook again const { result: result2 } = renderHook(() => useGasPrice()) - expect(result.current.maxFeePerGas?.toString()).toBe('21000000000') + // assert the hook is not loading (as a value exists) + expect(result.current[2]).toBe(false) + + expect(result.current[0]?.maxFeePerGas?.toString()).toBe('21000000000') // wait for the hook to fetch the gas price await act(async () => { await Promise.resolve() }) - expect(result2.current.maxFeePerGas?.toString()).toBe('22000000000') + // assert the hook is not loading + expect(result.current[2]).toBe(false) + + expect(result2.current[0]?.maxFeePerGas?.toString()).toBe('22000000000') }) }) diff --git a/src/hooks/useGasPrice.ts b/src/hooks/useGasPrice.ts index 056f78bc3f..2f7127e974 100644 --- a/src/hooks/useGasPrice.ts +++ b/src/hooks/useGasPrice.ts @@ -1,9 +1,7 @@ -import { useMemo } from 'react' import { BigNumber } from 'ethers' -import type { FeeData } from '@ethersproject/providers' import type { GasPrice, GasPriceOracle } from '@safe-global/safe-gateway-typescript-sdk' import { GAS_PRICE_TYPE } from '@safe-global/safe-gateway-typescript-sdk' -import useAsync from '@/hooks/useAsync' +import useAsync, { type AsyncResult } from '@/hooks/useAsync' import { useCurrentChain } from './useChains' import useIntervalCounter from './useIntervalCounter' import { useWeb3ReadOnly } from '../hooks/wallets/web3' @@ -54,41 +52,42 @@ const getGasPrice = async (gasPriceConfigs: GasPrice): Promise { +const useGasPrice = (): AsyncResult<{ + maxFeePerGas: BigNumber | undefined + maxPriorityFeePerGas: BigNumber | undefined +}> => { const chain = useCurrentChain() const gasPriceConfigs = chain?.gasPrice const [counter] = useIntervalCounter(REFRESH_DELAY) const provider = useWeb3ReadOnly() const isEIP1559 = !!chain && hasFeature(chain, FEATURES.EIP1559) - // Fetch gas price from oracles or get a fixed value - const [gasPrice] = useAsync( - () => { - if (gasPriceConfigs) { - return getGasPrice(gasPriceConfigs) + const [gasPrice, gasPriceError, gasPriceLoading] = useAsync( + async () => { + const [gasPrice, feeData] = await Promise.all([ + // Fetch gas price from oracles or get a fixed value + gasPriceConfigs ? getGasPrice(gasPriceConfigs) : undefined, + + // Fetch the gas fees from the blockchain itself + provider?.getFeeData(), + ]) + + // Prepare the return values + const maxFee = gasPrice || (isEIP1559 ? feeData?.maxFeePerGas : feeData?.gasPrice) || undefined + const maxPrioFee = (isEIP1559 && feeData?.maxPriorityFeePerGas) || undefined + + return { + maxFeePerGas: maxFee, + maxPriorityFeePerGas: maxPrioFee, } }, - [gasPriceConfigs, counter], + [gasPriceConfigs, provider, counter], false, ) - // Fetch the gas fees from the blockchain itself - const [feeData] = useAsync(() => provider?.getFeeData(), [provider, counter], false) + const isLoading = gasPriceLoading || (!gasPrice && !gasPriceError) - // Prepare the return values - const maxFee = gasPrice || (isEIP1559 ? feeData?.maxFeePerGas : feeData?.gasPrice) || undefined - const maxPrioFee = (isEIP1559 && feeData?.maxPriorityFeePerGas) || undefined - - return useMemo( - () => ({ - maxFeePerGas: maxFee, - maxPriorityFeePerGas: maxPrioFee, - }), - [maxFee, maxPrioFee], - ) + return [gasPrice, gasPriceError, isLoading] } export default useGasPrice diff --git a/src/services/tx/tx-sender/dispatch.ts b/src/services/tx/tx-sender/dispatch.ts index 3d1003c12a..0a05153bda 100644 --- a/src/services/tx/tx-sender/dispatch.ts +++ b/src/services/tx/tx-sender/dispatch.ts @@ -5,7 +5,7 @@ import { didReprice, didRevert } from '@/utils/ethers-utils' import type MultiSendCallOnlyEthersContract from '@safe-global/safe-ethers-lib/dist/src/contracts/MultiSendCallOnly/MultiSendCallOnlyEthersContract' import type { SpendingLimitTxParams } from '@/components/tx/modals/TokenTransferModal/ReviewSpendingLimitTx' import { getSpendingLimitContract } from '@/services/contracts/spendingLimitContracts' -import type { BigNumber, ContractTransaction } from 'ethers' +import type { ContractTransaction, PayableOverrides } from 'ethers' import type { RequestId } from '@safe-global/safe-apps-sdk' import proposeTx from '../proposeTransaction' import { txDispatch, TxEvent } from '../txEvents' @@ -174,7 +174,7 @@ export const dispatchBatchExecution = async ( onboard: OnboardAPI, chainId: SafeInfo['chainId'], safeAddress: string, - gasPrice?: BigNumber, + overrides?: PayableOverrides, ) => { const groupKey = multiSendTxData @@ -184,9 +184,7 @@ export const dispatchBatchExecution = async ( const wallet = await assertWalletChain(onboard, chainId) const provider = createWeb3(wallet.provider) - result = await multiSendContract.contract - .connect(provider.getSigner()) - .multiSend(multiSendTxData, { gasPrice: gasPrice?.toString() }) + result = await multiSendContract.contract.connect(provider.getSigner()).multiSend(multiSendTxData, overrides) txs.forEach(({ txId }) => { txDispatch(TxEvent.EXECUTING, { txId, groupKey }) diff --git a/src/utils/transactions.ts b/src/utils/transactions.ts index 0bba799811..f5dd81f4a2 100644 --- a/src/utils/transactions.ts +++ b/src/utils/transactions.ts @@ -30,9 +30,6 @@ import { Multi_send__factory } from '@/types/contracts' import { ethers } from 'ethers' import { type BaseTransaction } from '@safe-global/safe-apps-sdk' import { id } from 'ethers/lib/utils' -import chains from '@/config/chains' -import { WALLET_KEYS } from '@/hooks/wallets/consts' -import type { ConnectedWallet } from '@/services/onboard' export const makeTxFromDetails = (txDetails: TransactionDetails): Transaction => { const getMissingSigners = ({ @@ -269,16 +266,3 @@ export const decodeMultiSendTxs = (encodedMultiSendData: string): BaseTransactio return txs } - -/** - * Transactions are underpriced on Ledger if the gasPrice is not set - */ -export const shouldSetGasPrice = (chain: ChainInfo, wallet: ConnectedWallet): boolean => { - const POLYGON_CHAIN_ID = chains.matic - - const isEIP1559 = hasFeature(chain, FEATURES.EIP1559) - const isPolygon = chains[chain.shortName] === POLYGON_CHAIN_ID - const isLedger = wallet.label.toUpperCase() === WALLET_KEYS.LEDGER - - return !isEIP1559 && isLedger && isPolygon -} From a0da357b8d84c0da0707bffc9166e615e66cf3d0 Mon Sep 17 00:00:00 2001 From: iamacook Date: Wed, 5 Jul 2023 18:29:29 +0200 Subject: [PATCH 3/3] fix: mock --- .../create/steps/StatusStep/__tests__/useSafeCreation.test.ts | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/src/components/new-safe/create/steps/StatusStep/__tests__/useSafeCreation.test.ts b/src/components/new-safe/create/steps/StatusStep/__tests__/useSafeCreation.test.ts index 30c4ad2edb..4c3b3e4c58 100644 --- a/src/components/new-safe/create/steps/StatusStep/__tests__/useSafeCreation.test.ts +++ b/src/components/new-safe/create/steps/StatusStep/__tests__/useSafeCreation.test.ts @@ -114,9 +114,7 @@ describe('useSafeCreation', () => { }) it('should create a safe with no gas params if the gas estimation threw, there is no txHash and status is AWAITING', async () => { - jest - .spyOn(gasPrice, 'default') - .mockReturnValue([{ maxFeePerGas: undefined, maxPriorityFeePerGas: undefined }, undefined, false]) + jest.spyOn(gasPrice, 'default').mockReturnValue([undefined, Error('Error for testing'), false]) const createSafeSpy = jest.spyOn(logic, 'createNewSafe').mockReturnValue(Promise.resolve({} as Safe))