From 98590af9152ef2bfe56d9319b216155e4fd1b67d Mon Sep 17 00:00:00 2001 From: hemanthghs Date: Tue, 12 Nov 2024 10:44:10 +0530 Subject: [PATCH] feat: Implement offchain sequence management for multisig --- .../components/common/BroadCastTxn.tsx | 24 +++++++++++-- .../multisig/components/common/SignTxn.tsx | 33 ++++++++++++++--- .../multisig/components/common/TxnsCard.tsx | 16 +++++++-- .../multisig-account/DialogConfirmDelete.tsx | 2 +- .../multisig-account/Transactions.tsx | 35 ++++++++++++------- .../multisig-dashboard/RecentTransactions.tsx | 13 ++++--- .../multisig/utils/multisigSigning.ts | 5 +-- .../src/components/common/CustomButton.tsx | 2 +- .../store/features/multisig/multisigSlice.ts | 4 ++- frontend/src/types/multisig.d.ts | 1 + 10 files changed, 103 insertions(+), 32 deletions(-) diff --git a/frontend/src/app/(routes)/multisig/components/common/BroadCastTxn.tsx b/frontend/src/app/(routes)/multisig/components/common/BroadCastTxn.tsx index a8b0f7d95..82beaae44 100644 --- a/frontend/src/app/(routes)/multisig/components/common/BroadCastTxn.tsx +++ b/frontend/src/app/(routes)/multisig/components/common/BroadCastTxn.tsx @@ -12,6 +12,7 @@ import React, { useEffect } from 'react'; import { FAILED_TO_BROADCAST_ERROR } from '@/utils/errors'; import useVerifyAccount from '@/custom-hooks/useVerifyAccount'; import CustomButton from '@/components/common/CustomButton'; +import { useRouter } from 'next/navigation'; interface BroadCastTxnProps { txn: Txn; @@ -20,20 +21,33 @@ interface BroadCastTxnProps { pubKeys: MultisigAddressPubkey[]; chainID: string; isMember: boolean; + disableBroadcast?: boolean; + isOverview?: boolean; } const BroadCastTxn: React.FC = (props) => { - const { txn, multisigAddress, pubKeys, threshold, chainID, isMember } = props; + const { + txn, + multisigAddress, + pubKeys, + threshold, + chainID, + isMember, + disableBroadcast, + isOverview, + } = props; const dispatch = useAppDispatch(); const { getChainInfo } = useGetChainInfo(); const { address: walletAddress, restURLs: baseURLs, rpcURLs, + chainName, } = getChainInfo(chainID); const { isAccountVerified } = useVerifyAccount({ address: walletAddress, }); + const router = useRouter(); const updateTxnRes = useAppSelector( (state: RootState) => state.multisig.updateTxnRes @@ -78,9 +92,13 @@ const BroadCastTxn: React.FC = (props) => { { - broadcastTxn(); + if (isOverview) { + router.push(`/multisig/${chainName}/${multisigAddress}`); + } else { + broadcastTxn(); + } }} - btnDisabled={!isMember} + btnDisabled={!isMember || disableBroadcast} btnStyles="w-[115px]" /> ); diff --git a/frontend/src/app/(routes)/multisig/components/common/SignTxn.tsx b/frontend/src/app/(routes)/multisig/components/common/SignTxn.tsx index 784c6f61b..bf2ee18a5 100644 --- a/frontend/src/app/(routes)/multisig/components/common/SignTxn.tsx +++ b/frontend/src/app/(routes)/multisig/components/common/SignTxn.tsx @@ -1,4 +1,4 @@ -import { useAppDispatch } from '@/custom-hooks/StateHooks'; +import { useAppDispatch, useAppSelector } from '@/custom-hooks/StateHooks'; import useGetChainInfo from '@/custom-hooks/useGetChainInfo'; import { setVerifyDialogOpen, @@ -8,6 +8,7 @@ import { Txn } from '@/types/multisig'; import React from 'react'; import useVerifyAccount from '@/custom-hooks/useVerifyAccount'; import CustomButton from '@/components/common/CustomButton'; +import { useRouter } from 'next/navigation'; interface SignTxnProps { address: string; @@ -15,16 +16,33 @@ interface SignTxnProps { unSignedTxn: Txn; isMember: boolean; chainID: string; + isOverview?: boolean; } const SignTxn: React.FC = (props) => { - const { address, isMember, unSignedTxn, chainID } = props; + const { address, isMember, unSignedTxn, chainID, isOverview } = props; const dispatch = useAppDispatch(); const { getChainInfo } = useGetChainInfo(); - const { address: walletAddress, rpcURLs } = getChainInfo(chainID); + const { address: walletAddress, rpcURLs, chainName } = getChainInfo(chainID); const { isAccountVerified } = useVerifyAccount({ address: walletAddress, }); + const router = useRouter(); + + const txnsCount = useAppSelector((state) => state.multisig.txns.Count); + const getCount = (option: string) => { + let count = 0; + /* eslint-disable @typescript-eslint/no-explicit-any */ + txnsCount && + txnsCount.forEach((t: any) => { + if (t?.computed_status?.toLowerCase() === option.toLowerCase()) { + count = t?.count; + } + }); + + return count; + }; + const toBeBroadcastedCount = getCount('to-broadcast'); const signTheTx = async () => { if (!isAccountVerified()) { @@ -37,7 +55,8 @@ const SignTxn: React.FC = (props) => { multisigAddress: address, unSignedTxn, walletAddress, - rpcURLs + rpcURLs, + toBeBroadcastedCount, }) ); }; @@ -47,7 +66,11 @@ const SignTxn: React.FC = (props) => { btnText="Sign" btnDisabled={!isMember} btnOnClick={() => { - signTheTx(); + if (isOverview) { + router.push(`/multisig/${chainName}/${address}`); + } else { + signTheTx(); + } }} btnStyles="w-[115px]" /> diff --git a/frontend/src/app/(routes)/multisig/components/common/TxnsCard.tsx b/frontend/src/app/(routes)/multisig/components/common/TxnsCard.tsx index b10e1d786..4319fcdb9 100644 --- a/frontend/src/app/(routes)/multisig/components/common/TxnsCard.tsx +++ b/frontend/src/app/(routes)/multisig/components/common/TxnsCard.tsx @@ -40,6 +40,8 @@ export const TxnsCard = ({ isHistory, onViewError, allowRepeat, + disableBroadcast, + isOverview, }: { txn: Txn; currency: Currency; @@ -49,6 +51,8 @@ export const TxnsCard = ({ isHistory: boolean; onViewError?: (errMsg: string) => void; allowRepeat?: boolean; + disableBroadcast?: boolean; + isOverview?: boolean; }) => { const dispatch = useAppDispatch(); const { getChainInfo } = useGetChainInfo(); @@ -208,7 +212,7 @@ export const TxnsCard = ({
{isHistory || isReadyToBroadcast() - ? getTimeDifferenceToFutureDate(txn.last_updated, true) + ? getTimeDifferenceToFutureDate(txn.signed_at, true) : getTimeDifferenceToFutureDate(txn.created_at, true)}
@@ -268,6 +272,8 @@ export const TxnsCard = ({ threshold={threshold} chainID={chainID} isMember={isMember} + disableBroadcast={disableBroadcast} + isOverview={isOverview} /> ) : ( )} @@ -303,7 +310,12 @@ export const TxnsCard = ({ open={deleteDialogOpen} onClose={() => setDeleteDialogOpen(false)} title="Delete Transaction" - description=" Are you sure you want to delete the transaction ?" + description={ + 'Are you sure you want to delete the transaction?' + + (isReadyToBroadcast() + ? ' This action will require re-signing any already signed subsequent transactions.' + : '') + } onDelete={onDeleteTxn} loading={loading === TxStatus.PENDING} /> diff --git a/frontend/src/app/(routes)/multisig/components/multisig-account/DialogConfirmDelete.tsx b/frontend/src/app/(routes)/multisig/components/multisig-account/DialogConfirmDelete.tsx index 965458753..0b59195a6 100644 --- a/frontend/src/app/(routes)/multisig/components/multisig-account/DialogConfirmDelete.tsx +++ b/frontend/src/app/(routes)/multisig/components/multisig-account/DialogConfirmDelete.tsx @@ -54,7 +54,7 @@ const DialogConfirmDelete = ({ />
{title}
-
{description}
+
{description}
state.multisig.txns.Count) + const txnsCount = useAppSelector((state) => state.multisig.txns.Count); const txnsStatus = useAppSelector((state) => state.multisig.txns.status); const deleteTxnRes = useAppSelector((state) => state.multisig.deleteTxnRes); const signTxStatus = useAppSelector( @@ -172,21 +172,21 @@ const TransactionsFilters = ({ }: { txnsType: string; handleTxnsTypeChange: (type: string) => void; - /* eslint-disable-next-line @typescript-eslint/no-explicit-any */ + /* eslint-disable-next-line @typescript-eslint/no-explicit-any */ txnsCount: any; }) => { - const getCount = (option: string) => { let count = 0; - /* eslint-disable-next-line @typescript-eslint/no-explicit-any */ - txnsCount && txnsCount.forEach((t: any) => { - if (t?.computed_status?.toLowerCase() === option.toLowerCase()) { - count = t?.count - } - }) + /* eslint-disable @typescript-eslint/no-explicit-any */ + txnsCount && + txnsCount.forEach((t: any) => { + if (t?.computed_status?.toLowerCase() === option.toLowerCase()) { + count = t?.count; + } + }); - return count - } + return count; + }; return (
@@ -244,9 +244,19 @@ const TransactionsList = ({ setErrMsg(errMsg); setViewErrorDialogOpen(true); }; + const sortedTxns = [...txns].sort((a, b) => { + const dateA = new Date( + txnsType === 'to-broadcast' ? a.signed_at : a.created_at + ).getTime(); + const dateB = new Date( + txnsType === 'to-broadcast' ? b.signed_at : b.created_at + ).getTime(); + return txnsType === 'to-broadcast' ? dateA - dateB : dateB - dateA; + }); + return (
- {txns.map((txn, index) => ( + {sortedTxns.map((txn, index) => ( 0} /> ))} ))}
-
- -
+ {txns?.length > 1 ? ( +
+ +
+ ) : null}
); }; diff --git a/frontend/src/app/(routes)/multisig/utils/multisigSigning.ts b/frontend/src/app/(routes)/multisig/utils/multisigSigning.ts index 9c7d45b4a..f1a43051d 100644 --- a/frontend/src/app/(routes)/multisig/utils/multisigSigning.ts +++ b/frontend/src/app/(routes)/multisig/utils/multisigSigning.ts @@ -24,7 +24,8 @@ const signTransaction = async ( multisigAddress: string, unSignedTxn: Txn, walletAddress: string, - rpcURLs: string[] + rpcURLs: string[], + toBeBroadcastedCount: number ) => { try { window.wallet.defaultOptions = { @@ -47,7 +48,7 @@ const signTransaction = async ( const signerData = { accountNumber: multisigAcc?.accountNumber, - sequence: multisigAcc?.sequence, + sequence: multisigAcc?.sequence + toBeBroadcastedCount, chainId: chainID, }; diff --git a/frontend/src/components/common/CustomButton.tsx b/frontend/src/components/common/CustomButton.tsx index 500ba337d..1a6a47e48 100644 --- a/frontend/src/components/common/CustomButton.tsx +++ b/frontend/src/components/common/CustomButton.tsx @@ -30,7 +30,7 @@ const CustomButton = ({ }: CustomButtonProps) => { return (