Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(earn): support partial withdrawals #6128

Merged
merged 64 commits into from
Oct 18, 2024
Merged
Show file tree
Hide file tree
Changes from 44 commits
Commits
Show all changes
64 commits
Select commit Hold shift + click to select a range
6f1d0ee
chore: replace EarnDepositMode with EarnEnterMode
MuckT Oct 2, 2024
06ce84e
feat(earn): support partial withdrawals
MuckT Oct 2, 2024
d7769d2
Merge branch 'main' into tomm/act-1386
MuckT Oct 3, 2024
f5a612b
Merge branch 'main' into tomm/act-1386
MuckT Oct 4, 2024
3feb5f8
feat(earn): support partial withdrawals for earn pools
MuckT Oct 8, 2024
5821ff8
Merge branch 'main' into tomm/act-1386
MuckT Oct 8, 2024
dc155d2
fix(earn): rename EarnDepositMode to EarnEnterMode
MuckT Oct 8, 2024
a33d30c
fix(earn): rename EarnDepositMode to EarnEnterMode
MuckT Oct 8, 2024
4d01e46
fix: remove unused export
MuckT Oct 8, 2024
73c5067
fix(earn): amount TransactionWithdrawDetails display
MuckT Oct 9, 2024
364f551
test(earn): earn partial withdrawal flow
MuckT Oct 9, 2024
292d739
chore: add feature gate for partial withdrawals
MuckT Oct 9, 2024
87990f0
test(earn): partial withdrawal feature gate and navigation
MuckT Oct 9, 2024
6363e9f
docs: remove stray todo
MuckT Oct 9, 2024
71b13b9
Merge branch 'main' into tomm/act-1386
MuckT Oct 9, 2024
a58d40f
Merge branch 'main' into tomm/act-1386
MuckT Oct 9, 2024
345b535
Update test/values.ts
MuckT Oct 9, 2024
5fb98c5
refactor(earn): combine deposit and withdrawal hooks
MuckT Oct 9, 2024
96caa2e
fix(earn): use correct token amounts
MuckT Oct 11, 2024
9e8dcbf
Merge branch 'main' into tomm/act-1386
MuckT Oct 11, 2024
584acfb
fix(earn): disable continue when transaction is not possible
MuckT Oct 11, 2024
4196505
style(earn): use transactionToken instead of isWithdrawal ternaries
MuckT Oct 11, 2024
ebe3a2a
chore: PR feedback
MuckT Oct 11, 2024
1dbf295
style: naming
MuckT Oct 11, 2024
ad16ffa
fix(earn): use withdraw token and adjust display
MuckT Oct 12, 2024
7420841
Merge branch 'main' into tomm/act-1386
MuckT Oct 14, 2024
f5fecc5
fix: show correct withdrawal amounts
MuckT Oct 14, 2024
3879b52
fix: use correct fiat value display
MuckT Oct 14, 2024
9b29dfb
test(earn): show correct local amount
MuckT Oct 14, 2024
9ed4dfc
fix(earn): deposits should use input token
MuckT Oct 15, 2024
7b20e2d
fix(earn): use deposit token amounts in token input
MuckT Oct 15, 2024
6ed3c09
style(earn): use transaction token balance instead of pool balance
MuckT Oct 15, 2024
ca51c7d
test(earn): clean-up mock pool params
MuckT Oct 15, 2024
3150c45
Merge branch 'main' into tomm/act-1386
MuckT Oct 15, 2024
4bd8d62
Apply suggestions from code review
MuckT Oct 15, 2024
1f1db1e
style(earn): rename inputToken to depositToken in useTokens
MuckT Oct 15, 2024
819e297
fix(earn): set transaction token on token select for deposits
MuckT Oct 15, 2024
9c6a05d
fix(earn): display withdrawal text on fee bottom sheet
MuckT Oct 15, 2024
1da7fac
chore: pr feedback
MuckT Oct 15, 2024
686a01c
style: create reusable const balanceInTransactionToken
MuckT Oct 16, 2024
2681ce6
style: use input token directly
MuckT Oct 16, 2024
24c47be
fix(earn): use converted token amount in withdrawals
MuckT Oct 16, 2024
d6a1616
fix(earn): convert withdrawal amount correctly usd for analytics
MuckT Oct 16, 2024
6cef451
Merge branch 'main' into tomm/act-1386
MuckT Oct 16, 2024
da91f3d
fix(earn): remove duplicate condition
MuckT Oct 16, 2024
f85bed4
fix(earn): naming
MuckT Oct 16, 2024
c662e3b
docs: remove unnecessary comment
MuckT Oct 16, 2024
38853d8
style(earn): simplify dropdownEnabled
MuckT Oct 16, 2024
b989f05
fix(earn): use decimals from input token
MuckT Oct 16, 2024
2fe5293
style(earn): return single withdrawal and deposit token from useToken…
MuckT Oct 16, 2024
4af772d
style(earn): PR pair feedback
MuckT Oct 16, 2024
775a925
fix(earn): display correct amounts on input amount token toggle
MuckT Oct 16, 2024
176cb9d
Revert "fix(earn): display correct amounts on input amount token toggle"
MuckT Oct 16, 2024
2d87478
fix(earn): add inputToken to dependency array
MuckT Oct 17, 2024
8ccfbfc
chore(earn): update analytics for withdrawals
MuckT Oct 17, 2024
6924060
fix: analytics build errors
MuckT Oct 17, 2024
30c6ea9
test(earn): withdrawals from pools with high price per share
MuckT Oct 17, 2024
ea9f94e
Merge branch 'main' into tomm/act-1386
MuckT Oct 17, 2024
5deb486
Update src/earn/EarnEnterAmount.test.tsx
MuckT Oct 17, 2024
de1806b
chore(earn): use input token in analytics
MuckT Oct 17, 2024
bded4c8
test(earn): prepareWithdrawTransactions
MuckT Oct 17, 2024
ab9a0d7
test(earn-withdrawal): transaction preparation
MuckT Oct 18, 2024
6028676
Merge branch 'main' into tomm/act-1386
MuckT Oct 18, 2024
ed38faa
test(earn-withdrawal): remove duplicate mock clears
MuckT Oct 18, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 3 additions & 0 deletions locales/base/translation.json
Original file line number Diff line number Diff line change
Expand Up @@ -2554,8 +2554,10 @@
},
"enterAmount": {
"title": "How much would you like to deposit?",
"titleWithdraw": "How much would you like to withdraw?",
"deposit": "Deposit",
"fees": "Fees",
"available": "Available",
"swap": "Swap",
"earnUpToLabel": "You could earn up to:",
"rateLabel": "Rate (est.)",
Expand All @@ -2576,6 +2578,7 @@
"estNetworkFee": "Est. Network Fee",
"maxNetworkFee": "Max Network Fee",
"networkFeeDescription": "The network fee is required by the network to process the deposit transaction.",
"networkFeeDescriptionWithdrawal": "The network fee is required by the network to process the withdrawal transaction.",
"networkSwapFeeDescription": "The network fee is required by the network to process the deposit transactions. The {{appName}} fee of {{appFeePercentage}}% is charged for your use of our product.",
"appSwapFee": "{{appName}} Fee"
},
Expand Down
5 changes: 3 additions & 2 deletions src/analytics/Properties.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -54,7 +54,7 @@ import { ErrorMessages } from 'src/app/ErrorMessages'
import { AddAssetsActionType } from 'src/components/AddAssetsBottomSheet'
import { TokenPickerOrigin } from 'src/components/TokenBottomSheet'
import { DappSection } from 'src/dapps/types'
import { BeforeDepositActionName, EarnDepositMode, SerializableRewardsInfo } from 'src/earn/types'
import { BeforeDepositActionName, EarnEnterMode, SerializableRewardsInfo } from 'src/earn/types'
import { ProviderSelectionAnalyticsData } from 'src/fiatExchanges/types'
import { CICOFlow, FiatExchangeFlow, PaymentMethod } from 'src/fiatExchanges/utils'
import { HomeActionName, NotificationBannerCTATypes, NotificationType } from 'src/home/types'
Expand Down Expand Up @@ -590,6 +590,7 @@ interface SendEventsProperties {
tokenId: string
tokenAddress: string | null
networkId: NetworkId | null
mode?: EarnEnterMode
}
[SendEvents.swap_input_pressed]: {
swapToLocalAmount: boolean
Expand Down Expand Up @@ -1553,7 +1554,7 @@ export interface EarnCommonProperties {

interface EarnDepositProperties extends EarnCommonProperties {
depositTokenAmount: string
mode: EarnDepositMode
mode: EarnEnterMode
// the below are mainly for swap-deposit. For deposit, this would just be
// same as the depositTokenAmount and depositTokenId
fromTokenAmount: string
Expand Down
4 changes: 2 additions & 2 deletions src/earn/EarnDepositBottomSheet.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ import { LabelWithInfo } from 'src/components/LabelWithInfo'
import TokenDisplay from 'src/components/TokenDisplay'
import { depositStatusSelector } from 'src/earn/selectors'
import { depositStart } from 'src/earn/slice'
import { EarnDepositMode } from 'src/earn/types'
import { EarnEnterMode } from 'src/earn/types'
import {
getSwapToAmountInDecimals,
getTotalYieldRate,
Expand Down Expand Up @@ -50,7 +50,7 @@ export default function EarnDepositBottomSheet({
inputTokenId: string
inputAmount: BigNumber
pool: EarnPosition
mode: EarnDepositMode
mode: EarnEnterMode
swapTransaction?: SwapTransaction
}) {
const { t } = useTranslation()
Expand Down
103 changes: 92 additions & 11 deletions src/earn/EarnEnterAmount.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ import { Provider } from 'react-redux'
import AppAnalytics from 'src/analytics/AppAnalytics'
import { EarnEvents } from 'src/analytics/Events'
import EarnEnterAmount from 'src/earn/EarnEnterAmount'
import { usePrepareDepositTransactions } from 'src/earn/prepareTransactions'
import { usePrepareTransactions } from 'src/earn/prepareTransactions'
import { CICOFlow } from 'src/fiatExchanges/utils'
import { navigate } from 'src/navigator/NavigationService'
import { Screens } from 'src/navigator/Screens'
Expand All @@ -21,6 +21,7 @@ import networkConfig from 'src/web3/networkConfig'
import MockedNavigator from 'test/MockedNavigator'
import { createMockStore } from 'test/utils'
import {
mockAaveArbUsdcTokenId,
mockAccount,
mockArbArbTokenId,
mockArbEthTokenId,
Expand Down Expand Up @@ -111,7 +112,8 @@ const mockSwapTransaction: SwapTransaction = {
const store = createMockStore({
tokens: {
tokenBalances: {
[mockArbUsdcTokenId]: {
...mockTokenBalances,
mockArbUsdcTokenId: {
...mockTokenBalances[mockArbUsdcTokenId],
balance: '10',
},
Expand All @@ -124,6 +126,10 @@ const store = createMockStore({
minimumAppVersionToSwap: '1.0.0',
balance: '1',
},
mockAaveArbUsdcTokenId: {
...mockTokenBalances[mockAaveArbUsdcTokenId],
balance: '10',
},
},
},
})
Expand All @@ -140,7 +146,7 @@ describe('EarnEnterAmount', () => {
.mocked(getNumberFormatSettings)
.mockReturnValue({ decimalSeparator: '.', groupingSeparator: ',' })
store.clearActions()
jest.mocked(usePrepareDepositTransactions).mockReturnValue({
jest.mocked(usePrepareTransactions).mockReturnValue({
prepareTransactionsResult: undefined,
refreshPreparedTransactions: refreshPreparedTransactionsSpy,
clearPreparedTransactions: jest.fn(),
Expand Down Expand Up @@ -187,11 +193,12 @@ describe('EarnEnterAmount', () => {
hooksApiUrl: networkConfig.hooksApiUrl,
feeCurrencies: mockFeeCurrencies,
shortcutId: 'deposit',
useMax: false,
})
})

it('should show tx details and handle navigating to the deposit bottom sheet', async () => {
jest.mocked(usePrepareDepositTransactions).mockReturnValue({
jest.mocked(usePrepareTransactions).mockReturnValue({
prepareTransactionsResult: {
prepareTransactionsResult: mockPreparedTransaction,
swapTransaction: undefined,
Expand Down Expand Up @@ -262,6 +269,7 @@ describe('EarnEnterAmount', () => {
const store = createMockStore({
tokens: {
tokenBalances: {
...mockTokenBalances,
[mockArbUsdcTokenId]: {
...mockTokenBalances[mockArbUsdcTokenId],
balance: '10',
Expand Down Expand Up @@ -314,11 +322,12 @@ describe('EarnEnterAmount', () => {
hooksApiUrl: networkConfig.hooksApiUrl,
feeCurrencies: mockFeeCurrencies,
shortcutId: 'swap-deposit',
useMax: false,
})
})

it('should show tx details and handle navigating to the deposit bottom sheet', async () => {
jest.mocked(usePrepareDepositTransactions).mockReturnValue({
jest.mocked(usePrepareTransactions).mockReturnValue({
prepareTransactionsResult: {
prepareTransactionsResult: mockPreparedTransaction,
swapTransaction: mockSwapTransaction,
Expand Down Expand Up @@ -372,9 +381,80 @@ describe('EarnEnterAmount', () => {
})
})

describe('withdraw', () => {
// Pool balance should be set to determine available withdrawal amount
const withdrawParams = { ...params, mode: 'withdraw' }

beforeEach(() => {
jest.clearAllMocks()
MuckT marked this conversation as resolved.
Show resolved Hide resolved
jest.mocked(usePrepareTransactions).mockReturnValue({
prepareTransactionsResult: {
prepareTransactionsResult: mockPreparedTransaction,
swapTransaction: undefined,
},
refreshPreparedTransactions: jest.fn(),
clearPreparedTransactions: jest.fn(),
prepareTransactionError: undefined,
isPreparingTransactions: false,
})
})
it('should show only the deposit token and not include the token dropdown', async () => {
const { getByTestId, queryByTestId } = render(
<Provider store={store}>
<MockedNavigator component={EarnEnterAmount} params={withdrawParams} />
</Provider>
)

expect(getByTestId('EarnEnterAmount/TokenSelect')).toBeTruthy()
expect(getByTestId('EarnEnterAmount/TokenSelect')).toHaveTextContent('USDC')
expect(getByTestId('EarnEnterAmount/TokenSelect')).toBeDisabled()
expect(queryByTestId('downArrowIcon')).toBeFalsy()
})

it('should show tx details for withdrawal', async () => {
const { getByTestId, getByText } = render(
<Provider store={store}>
<MockedNavigator component={EarnEnterAmount} params={withdrawParams} />
</Provider>
)

fireEvent.changeText(getByTestId('EarnEnterAmount/TokenAmountInput'), '8')

await waitFor(() => expect(getByText('earnFlow.enterAmount.continue')).not.toBeDisabled())

expect(getByTestId('EarnEnterAmount/Withdraw/Crypto')).toBeTruthy()
expect(getByTestId('EarnEnterAmount/Withdraw/Crypto')).toHaveTextContent('11.00 USDC')

expect(getByTestId('EarnEnterAmount/Withdraw/Fiat')).toBeTruthy()
expect(getByTestId('EarnEnterAmount/Withdraw/Fiat')).toBeTruthy()
expect(getByTestId('EarnEnterAmount/Withdraw/Fiat')).toHaveTextContent('₱14.63')

expect(getByTestId('EarnEnterAmount/Fees')).toBeTruthy()
expect(getByTestId('EarnEnterAmount/Fees')).toHaveTextContent('₱0.012')

fireEvent.press(getByText('earnFlow.enterAmount.continue'))

await waitFor(() => expect(AppAnalytics.track).toHaveBeenCalledTimes(1))
expect(AppAnalytics.track).toHaveBeenCalledWith(EarnEvents.earn_enter_amount_continue_press, {
amountEnteredIn: 'token',
amountInUsd: '8.00',
networkId: NetworkId['arbitrum-sepolia'],
depositTokenId: mockArbUsdcTokenId,
providerId: mockEarnPositions[0].appId,
poolId: mockEarnPositions[0].positionId,
fromTokenId: 'arbitrum-sepolia:0x460b97bd498e1157530aeb3086301d5225b91216',
fromTokenAmount: '8',
depositTokenAmount: '8',
mode: 'withdraw',
})

//TODO(ACT-1389): check navigation to withdrawal confirmation screen
})
})

// tests independent of deposit / swap-deposit
it('should show a warning and not allow the user to continue if they input an amount greater than balance', async () => {
jest.mocked(usePrepareDepositTransactions).mockReturnValue({
jest.mocked(usePrepareTransactions).mockReturnValue({
prepareTransactionsResult: {
prepareTransactionsResult: mockPreparedTransaction,
swapTransaction: undefined,
Expand All @@ -397,7 +477,7 @@ describe('EarnEnterAmount', () => {
})

it('should show loading spinner when preparing transaction', async () => {
jest.mocked(usePrepareDepositTransactions).mockReturnValue({
jest.mocked(usePrepareTransactions).mockReturnValue({
prepareTransactionsResult: undefined,
refreshPreparedTransactions: jest.fn(),
clearPreparedTransactions: jest.fn(),
Expand Down Expand Up @@ -443,6 +523,7 @@ describe('EarnEnterAmount', () => {
const mockStore = createMockStore({
tokens: {
tokenBalances: {
...mockTokenBalances,
[mockArbUsdcTokenId]: {
...mockTokenBalances[mockArbUsdcTokenId],
balance: '100000.42',
Expand All @@ -469,7 +550,7 @@ describe('EarnEnterAmount', () => {
})

it('should track analytics and navigate correctly when tapping cta to add gas', async () => {
jest.mocked(usePrepareDepositTransactions).mockReturnValue({
jest.mocked(usePrepareTransactions).mockReturnValue({
prepareTransactionsResult: {
prepareTransactionsResult: mockPreparedTransactionNotEnough,
swapTransaction: undefined,
Expand Down Expand Up @@ -506,7 +587,7 @@ describe('EarnEnterAmount', () => {
})

it('should show the FeeDetailsBottomSheet when the user taps the fee details icon', async () => {
jest.mocked(usePrepareDepositTransactions).mockReturnValue({
jest.mocked(usePrepareTransactions).mockReturnValue({
prepareTransactionsResult: {
prepareTransactionsResult: mockPreparedTransaction,
swapTransaction: undefined,
Expand All @@ -531,7 +612,7 @@ describe('EarnEnterAmount', () => {
})

it('should show swap fees on the FeeDetailsBottomSheet when swap transaction is present', async () => {
jest.mocked(usePrepareDepositTransactions).mockReturnValue({
jest.mocked(usePrepareTransactions).mockReturnValue({
prepareTransactionsResult: {
prepareTransactionsResult: mockPreparedTransaction,
swapTransaction: mockSwapTransaction,
Expand Down Expand Up @@ -562,7 +643,7 @@ describe('EarnEnterAmount', () => {
})

it('should display swap bottom sheet when the user taps the swap details icon', async () => {
jest.mocked(usePrepareDepositTransactions).mockReturnValue({
jest.mocked(usePrepareTransactions).mockReturnValue({
prepareTransactionsResult: {
prepareTransactionsResult: mockPreparedTransaction,
swapTransaction: mockSwapTransaction,
Expand Down
Loading
Loading