From 26a70e96fb03033cec8caebf481325c596e7be16 Mon Sep 17 00:00:00 2001 From: Jared Vu Date: Thu, 13 Jun 2024 15:58:26 -0700 Subject: [PATCH] feat(rebalance-wallet): rebalance wallet if usdcBalance is low (#672) --- src/constants/account.ts | 1 + src/hooks/useSubaccount.tsx | 48 +++++++++++++++++++-------------- src/views/menus/AccountMenu.tsx | 29 +++++++++++++++++--- 3 files changed, 54 insertions(+), 24 deletions(-) diff --git a/src/constants/account.ts b/src/constants/account.ts index 9ef1dfed0..20db9640d 100644 --- a/src/constants/account.ts +++ b/src/constants/account.ts @@ -51,6 +51,7 @@ export type Hdkey = { }; export const AMOUNT_RESERVED_FOR_GAS_USDC = 0.5; +export const AMOUNT_USDC_BEFORE_REBALANCE = 0.05; /** * @description The number of parentSubaccounts: 0 - 127, 128 is the first childSubaccount diff --git a/src/hooks/useSubaccount.tsx b/src/hooks/useSubaccount.tsx index e83445d38..1570d372e 100644 --- a/src/hooks/useSubaccount.tsx +++ b/src/hooks/useSubaccount.tsx @@ -22,7 +22,7 @@ import type { HumanReadableTriggerOrdersPayload, ParsingError, } from '@/constants/abacus'; -import { AMOUNT_RESERVED_FOR_GAS_USDC } from '@/constants/account'; +import { AMOUNT_RESERVED_FOR_GAS_USDC, AMOUNT_USDC_BEFORE_REBALANCE } from '@/constants/account'; import { STRING_KEYS } from '@/constants/localization'; import { QUANTUM_MULTIPLIER } from '@/constants/numbers'; import { TradeTypes } from '@/constants/trade'; @@ -240,7 +240,7 @@ const useSubaccountContext = ({ localDydxWallet }: { localDydxWallet?: LocalWall } }, }), - [compositeClient] + [compositeClient, usdcDecimals] ); const [subaccountNumber] = useState(0); @@ -259,21 +259,29 @@ const useSubaccountContext = ({ localDydxWallet }: { localDydxWallet?: LocalWall useEffect(() => { dispatch(setSubaccount(undefined)); dispatch(setHistoricalPnl([])); - }, [dydxAddress]); + }, [dispatch, dydxAddress]); // ------ Deposit/Withdraw Methods ------ // - const depositFunds = useCallback( + const rebalanceWalletFunds = useCallback( async (balance: AccountBalance) => { - if (!localDydxWallet) return; - - const amount = parseFloat(balance.amount) - AMOUNT_RESERVED_FOR_GAS_USDC; - - if (amount > 0) { - const newSubaccountClient = new SubaccountClient(localDydxWallet, 0); - await depositToSubaccount({ amount, subaccountClient: newSubaccountClient }); + if (!subaccountClient) return; + const balanceAmount = parseFloat(balance.amount); + const shouldDeposit = balanceAmount - AMOUNT_RESERVED_FOR_GAS_USDC > 0; + const shouldWithdraw = balanceAmount - AMOUNT_USDC_BEFORE_REBALANCE <= 0; + + if (shouldDeposit) { + await depositToSubaccount({ + amount: balanceAmount - AMOUNT_RESERVED_FOR_GAS_USDC, + subaccountClient, + }); + } else if (shouldWithdraw) { + await withdrawFromSubaccount({ + amount: AMOUNT_RESERVED_FOR_GAS_USDC - balanceAmount, + subaccountClient, + }); } }, - [localDydxWallet, depositToSubaccount] + [subaccountClient, depositToSubaccount, withdrawFromSubaccount] ); const balances = useAppSelector(getBalances, shallowEqual); @@ -281,9 +289,9 @@ const useSubaccountContext = ({ localDydxWallet }: { localDydxWallet?: LocalWall useEffect(() => { if (usdcCoinBalance) { - depositFunds(usdcCoinBalance); + rebalanceWalletFunds(usdcCoinBalance); } - }, [usdcCoinBalance]); + }, [usdcCoinBalance, rebalanceWalletFunds]); const deposit = useCallback( async (amount: number) => { @@ -318,7 +326,7 @@ const useSubaccountContext = ({ localDydxWallet }: { localDydxWallet?: LocalWall ? transferFromSubaccountToAddress({ subaccountClient, amount, recipient }) : transferNativeToken({ subaccountClient, amount, recipient, memo }))) as IndexedTx; }, - [subaccountClient, transferFromSubaccountToAddress, transferNativeToken] + [subaccountClient, transferFromSubaccountToAddress, transferNativeToken, usdcDenom] ); const sendSquidWithdraw = useCallback( @@ -374,7 +382,7 @@ const useSubaccountContext = ({ localDydxWallet }: { localDydxWallet?: LocalWall const subaccountTransferPayload = abacusStateManager.adjustIsolatedMarginOfPosition(callback); return subaccountTransferPayload; }, - [subaccountClient] + [] ); // ------ Faucet Methods ------ // @@ -444,7 +452,7 @@ const useSubaccountContext = ({ localDydxWallet }: { localDydxWallet?: LocalWall return placeOrderParams; }, - [subaccountClient] + [dispatch] ); const closePosition = useCallback( @@ -486,7 +494,7 @@ const useSubaccountContext = ({ localDydxWallet }: { localDydxWallet?: LocalWall dispatch(cancelOrderSubmitted(orderId)); abacusStateManager.cancelOrder(orderId, callback); }, - [subaccountClient] + [dispatch] ); // ------ Trigger Orders Methods ------ // @@ -557,7 +565,7 @@ const useSubaccountContext = ({ localDydxWallet }: { localDydxWallet?: LocalWall return triggerOrderParams; }, - [subaccountClient] + [dispatch] ); const { newMarketProposal } = useGovernanceVariables(); @@ -583,7 +591,7 @@ const useSubaccountContext = ({ localDydxWallet }: { localDydxWallet?: LocalWall return response; }, - [compositeClient, localDydxWallet] + [compositeClient, localDydxWallet, newMarketProposal] ); // ------ Staking Methods ------ // diff --git a/src/views/menus/AccountMenu.tsx b/src/views/menus/AccountMenu.tsx index cc9016eb3..1141711b2 100644 --- a/src/views/menus/AccountMenu.tsx +++ b/src/views/menus/AccountMenu.tsx @@ -15,6 +15,7 @@ import { type StringGetterFunction, } from '@/constants/localization'; import { isDev } from '@/constants/networks'; +import { SMALL_USD_DECIMALS, USD_DECIMALS } from '@/constants/numbers'; import { DydxChainAsset, WalletType, wallets } from '@/constants/wallets'; import { useAccountBalance } from '@/hooks/useAccountBalance'; @@ -60,7 +61,7 @@ export const AccountMenu = () => { const onboardingState = useAppSelector(getOnboardingState); const { freeCollateral } = useAppSelector(getSubaccount, shallowEqual) ?? {}; - const { nativeTokenBalance } = useAccountBalance(); + const { nativeTokenBalance, usdcBalance } = useAccountBalance(); const { usdcLabel, chainTokenLabel } = useTokenConfigs(); const theme = useAppSelector(getAppTheme); @@ -71,8 +72,6 @@ export const AccountMenu = () => { const { showMfaEnrollmentModal } = useMfaEnrollment(); - const usdcBalance = freeCollateral?.current ?? 0; - const onRecoverKeys = () => { dispatch(openDialog({ type: DialogTypes.Onboarding })); }; @@ -169,6 +168,24 @@ export const AccountMenu = () => { stringGetter={stringGetter} /> + {isDev && ( +
+
+ <$label> + {stringGetter({ + key: STRING_KEYS.WALLET_BALANCE, + params: { ASSET: usdcLabel }, + })} + + + <$BalanceOutput + type={OutputType.Asset} + value={usdcBalance} + fractionDigits={SMALL_USD_DECIMALS} + /> +
+
+ )}
<$label> @@ -178,7 +195,11 @@ export const AccountMenu = () => { })} - <$BalanceOutput type={OutputType.Asset} value={usdcBalance} fractionDigits={2} /> + <$BalanceOutput + type={OutputType.Asset} + value={freeCollateral?.current ?? 0} + fractionDigits={USD_DECIMALS} + />