diff --git a/apps/frontend/pages/accounting.tsx b/apps/frontend/pages/accounting.tsx index 749d088a..2755ffd7 100644 --- a/apps/frontend/pages/accounting.tsx +++ b/apps/frontend/pages/accounting.tsx @@ -1,4 +1,3 @@ -/* eslint-disable no-useless-computed-key */ import { Button, Flex, @@ -9,22 +8,18 @@ import { TabPanels, Tabs, } from '@raidguild/design-system'; -import { useAccounting, useMemberList } from '@raidguild/dm-hooks'; import { - IMember, - ITokenBalanceLineItem, - IVaultTransaction, -} from '@raidguild/dm-types'; -import { - exportToCsv, - formatDate, - REGEX_ETH_ADDRESS, -} from '@raidguild/dm-utils'; + useAccountingV3, + useAccountingV2, + useFormattedData, + useMemberList, +} from '@raidguild/dm-hooks'; +import { exportToCsv } from '@raidguild/dm-utils'; import _ from 'lodash'; import { useSession } from 'next-auth/react'; import { NextSeo } from 'next-seo'; import Papa from 'papaparse'; -import { useCallback, useMemo } from 'react'; +import { useCallback } from 'react'; import BalancesTable from '../components/BalancesTable'; import SiteLayout from '../components/SiteLayout'; @@ -34,76 +29,27 @@ import TransactionsTable from '../components/TransactionsTable'; export const Accounting = () => { const { data: session } = useSession(); const token = _.get(session, 'token'); - const { data, loading, error } = useAccounting({ + const { + data: dataFromMolochV2, + loading, + error, + } = useAccountingV2({ token, }); + const { data: dataFromMolochV3 } = useAccountingV3(); const { data: memberData } = useMemberList({ token, limit: 1000, }); - const { balances, spoils, transactions, tokenPrices } = data; - - const members = useMemo(() => { - const memberArray = _.flatten( - _.get(memberData, 'pages') - ) as unknown as IMember[]; - return _.keyBy(memberArray, (m: IMember) => m.ethAddress?.toLowerCase()); - }, [memberData]); - - const withPrices = useCallback( - (items: T[]) => - items.map((t) => { - const formattedDate = formatDate(t.date); - const tokenSymbol = t.tokenSymbol?.toLowerCase(); - if ( - tokenPrices[tokenSymbol] && - tokenPrices[tokenSymbol][formattedDate] - ) { - return { - ...t, - priceConversion: tokenPrices[tokenSymbol][formattedDate], - }; - } - if (tokenSymbol.includes('xdai')) { - return { - ...t, - priceConversion: 1, - }; - } - return t; - }), - [tokenPrices] - ); + const { balances, spoils, transactions, tokenPrices } = dataFromMolochV2; - const balancesWithPrices = useMemo( - () => withPrices(balances), - [balances, withPrices] - ); - - const transactionsWithPrices = useMemo( - () => withPrices(transactions), - [transactions, withPrices] - ); - - const transactionsWithPricesAndMembers = useMemo( - () => - transactionsWithPrices.map((t) => { - const ethAddress = t.proposalApplicant.toLowerCase(); - const m = members[ethAddress]; - const memberLink = m?.ethAddress.match(REGEX_ETH_ADDRESS) - ? `/members/${ethAddress}` - : undefined; - - return { - ...t, - memberLink, - memberName: m?.name, - memberEnsName: m?.ensName, - }; - }), - [transactionsWithPrices, members] - ); + const { + members, + balancesWithPrices, + transactionsWithPrices, + transactionsWithPricesAndMembers, + } = useFormattedData(memberData, balances, transactions, tokenPrices); const onExportCsv = useCallback( (type: 'transactions' | 'balances' | 'spoils') => { @@ -212,52 +158,123 @@ export const Accounting = () => { - - - - + + + V3 (current) + + + V2 + + + + + + + + + + +
This is the placeholder for v3 balances data.
+
+
+
- - - - + + + V3 (current) + + + V2 + + + + + + + + + + +
This is the placeholder for v3 transactions data.
+
+
+
- - - - + + + V3 (current) + + + V2 + + + + + + + + + + +
This is the placeholder for v3 spoils data.
+
+
+
diff --git a/libs/dm-hooks/src/index.ts b/libs/dm-hooks/src/index.ts index ebc2d630..0cf07cb4 100644 --- a/libs/dm-hooks/src/index.ts +++ b/libs/dm-hooks/src/index.ts @@ -1,4 +1,5 @@ -export { default as useAccounting } from './useAccounting'; +export { default as useAccountingV2 } from './useAccountingV2'; +export { default as useAccountingV3 } from './useAccountingV3'; export { default as useApplicationDetail } from './useApplicationDetail'; export { default as useApplicationList, @@ -15,6 +16,7 @@ export { useContacts } from './useContacts'; export { default as useContactUpdate } from './useContactUpdate'; export { default as useDashboardList } from './useDashboardList'; export { default as useDefaultTitle } from './useDefaultTitle'; +export { default as useFormattedData } from './useFormattedData'; export * from './useLinks'; export { default as useLinksUpdate } from './useLinksUpdate'; export { default as useMemberCreate } from './useMemberCreate'; diff --git a/libs/dm-hooks/src/useAccounting.ts b/libs/dm-hooks/src/useAccountingV2.ts similarity index 99% rename from libs/dm-hooks/src/useAccounting.ts rename to libs/dm-hooks/src/useAccountingV2.ts index ab5fdf2f..4e99af1c 100644 --- a/libs/dm-hooks/src/useAccounting.ts +++ b/libs/dm-hooks/src/useAccountingV2.ts @@ -343,7 +343,7 @@ const formatSpoils = async ( return spoils.sort((a, b) => b.date.getTime() - a.date.getTime()); }; -export const useAccounting = ({ token }: { token: string }) => { +export const useAccountingV2 = ({ token }: { token: string }) => { const [transactions, setTransactions] = useState>( [] ); @@ -453,4 +453,4 @@ export const useAccounting = ({ token }: { token: string }) => { }; }; -export default useAccounting; +export default useAccountingV2; diff --git a/libs/dm-hooks/src/useAccountingV3.ts b/libs/dm-hooks/src/useAccountingV3.ts new file mode 100644 index 00000000..b7d54726 --- /dev/null +++ b/libs/dm-hooks/src/useAccountingV3.ts @@ -0,0 +1,82 @@ +import { da } from 'date-fns/locale'; +import { useState, useEffect } from 'react'; + +import { getAddress } from 'viem'; + + +export const transformTokenBalances = ( + tokenBalanceRes: any, + safeAddress: string +): any => { + const fiatTotal = tokenBalanceRes.reduce( + (sum: number, balance: any): number => { + sum += Number(balance.fiatBalance); + return sum; + }, + 0 + ); + + return { safeAddress, tokenBalances: tokenBalanceRes, fiatTotal }; +}; + +export const listTokenBalances = async ({ + safeAddress, +}: { + safeAddress: string; +}): Promise => { + const url = 'https://safe-transaction-gnosis-chain.safe.global/api/v1' + + if (!url) { + return { + error: 'Error fetching token balances. Please try again.' + }; + } + + try { + const res = await fetch( + `${url}/safes/${safeAddress}/balances/usd/` + ); + const data = await res.json(); + console.log(data) + // + return { data: transformTokenBalances(data, safeAddress) }; + } catch (err) { + return { + error: 'Error fetching token balances. Please try again.' + }; + } +}; + +// this is a dummy hook +const useAccountingV3 = () => { + const [data, setData] = useState(null); + const [loading, setLoading] = useState(true); + const [error, setError] = useState(null); + + useEffect(() => { + const fetchData = async () => { + const checksum = getAddress('0x181ebdb03cb4b54f4020622f1b0eacd67a8c63ac'); + console.log(checksum) + const { data, error } = await listTokenBalances({ + safeAddress: checksum, + }); + + if (error) { + setError(error); + console.log('error', error) + } else { + setData(data); + console.log('data', data); + } + + setLoading(false); + }; + + fetchData(); + console.log('fetching data', data); + }, []) + + return { data: null, loading: false, error: null }; +}; + +export default useAccountingV3; diff --git a/libs/dm-hooks/src/useFormattedData.ts b/libs/dm-hooks/src/useFormattedData.ts new file mode 100644 index 00000000..4fd93780 --- /dev/null +++ b/libs/dm-hooks/src/useFormattedData.ts @@ -0,0 +1,87 @@ +import { + IMappedTokenPrice, + IMember, + ITokenBalanceLineItem, + IVaultTransaction, +} from '@raidguild/dm-types'; +import { formatDate, REGEX_ETH_ADDRESS } from '@raidguild/dm-utils'; +import { InfiniteData } from '@tanstack/react-query'; +import _ from 'lodash'; +import { useCallback, useMemo } from 'react'; + +const useFormattedData = ( + memberData: InfiniteData, + balances: ITokenBalanceLineItem[], + transactions: IVaultTransaction[], + tokenPrices: IMappedTokenPrice +) => { + const members = useMemo(() => { + const memberArray = _.flatten( + _.get(memberData, 'pages') + ) as unknown as IMember[]; + return _.keyBy(memberArray, (m: IMember) => m.ethAddress?.toLowerCase()); + }, [memberData]); + + const withPrices = useCallback( + (items: T[]) => + items.map((t) => { + const formattedDate = formatDate(t.date); + const tokenSymbol = t.tokenSymbol?.toLowerCase(); + if ( + tokenPrices[tokenSymbol] && + tokenPrices[tokenSymbol][formattedDate] + ) { + return { + ...t, + priceConversion: tokenPrices[tokenSymbol][formattedDate], + }; + } + if (tokenSymbol.includes('xdai')) { + return { + ...t, + priceConversion: 1, + }; + } + return t; + }), + [tokenPrices] + ); + + const balancesWithPrices = useMemo( + () => withPrices(balances), + [balances, withPrices] + ); + + const transactionsWithPrices = useMemo( + () => withPrices(transactions), + [transactions, withPrices] + ); + + const transactionsWithPricesAndMembers = useMemo( + () => + transactionsWithPrices.map((t) => { + const ethAddress = t.proposalApplicant.toLowerCase(); + const m = members[ethAddress]; + const memberLink = m?.ethAddress.match(REGEX_ETH_ADDRESS) + ? `/members/${ethAddress}` + : undefined; + + return { + ...t, + memberLink, + memberName: m?.name, + memberEnsName: m?.ensName, + }; + }), + [transactionsWithPrices, members] + ); + + return { + members, + balancesWithPrices, + transactionsWithPrices, + transactionsWithPricesAndMembers, + }; +}; + +export default useFormattedData;