Skip to content

Commit

Permalink
Merge pull request #607 from valory-xyz/mohan/memeooorr-manage-wallet
Browse files Browse the repository at this point in the history
feat: manage wallet for multi-agent
  • Loading branch information
mohandast52 authored Dec 18, 2024
2 parents 1baced9 + 32f3d02 commit 7c2ba1c
Show file tree
Hide file tree
Showing 10 changed files with 189 additions and 114 deletions.
7 changes: 2 additions & 5 deletions frontend/components/AddressLink.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -7,16 +7,13 @@ import { truncateAddress } from '@/utils/truncate';
type AddressLinkProps = {
address?: Address;
hideLinkArrow?: boolean;

// TODO: mark as required once balance breakdown is updated.
// and remove the default value
middlewareChain?: MiddlewareChain;
middlewareChain: MiddlewareChain;
};

export const AddressLink = ({
address,
hideLinkArrow = false,
middlewareChain = MiddlewareChain.GNOSIS,
middlewareChain,
}: AddressLinkProps) => {
if (!address) return null;
if (!middlewareChain) return null;
Expand Down
4 changes: 0 additions & 4 deletions frontend/components/SettingsPage/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@ import { CustomAlert } from '../Alert';
import { CardTitle } from '../Card/CardTitle';
import { CardSection } from '../styled/CardSection';
import { AddBackupWalletPage } from './AddBackupWalletPage';
import { DebugInfoSection } from './DebugInfoSection';

const { Text, Paragraph } = Typography;

Expand Down Expand Up @@ -161,9 +160,6 @@ const SettingsMain = () => {
<Text strong>Backup wallet</Text>
{walletBackup}
</CardSection>

{/* Debug info */}
<DebugInfoSection />
</Card>
);
};
15 changes: 12 additions & 3 deletions frontend/components/YourWalletPage/Titles.tsx
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import { Flex, Typography } from 'antd';

import { MiddlewareChain } from '@/client';
import { InfoTooltip } from '@/components/InfoTooltip';
import { TokenSymbol } from '@/enums/Token';
import { Address } from '@/types/Address';
Expand All @@ -8,11 +9,16 @@ import { AddressLink } from '../AddressLink';

const { Paragraph, Text, Title } = Typography;

type SignerTitleProps = { signerText: string; signerAddress?: Address };
type SignerTitleProps = {
signerText: string;
signerAddress: Address;
middlewareChain: MiddlewareChain;
};

export const SignerTitle = ({
signerText,
signerAddress,
middlewareChain,
}: SignerTitleProps) => (
<>
Signer&nbsp;
Expand All @@ -28,7 +34,10 @@ export const SignerTitle = ({
<Paragraph className="text-sm m-0">
<Flex gap={4} vertical>
<Text className="text-sm">{signerText}</Text>
<AddressLink address={signerAddress} />
<AddressLink
address={signerAddress}
middlewareChain={middlewareChain}
/>
</Flex>
</Paragraph>
</InfoTooltip>
Expand Down Expand Up @@ -58,7 +67,7 @@ export const NativeTokenTitle = ({ symbol }: { symbol: TokenSymbol }) => (
</Title>
&nbsp;
<InfoTooltip>
{/* TODO: address multi-agent tooltip, specfic to agent config */}
{/* TODO: address multi-agent tooltip, specific to agent config */}
<Paragraph className="text-sm m-0">
{symbol} is used by the agent to engage in prediction markets. This
amount will fluctuate based on your agent&apos;s performance.
Expand Down
7 changes: 6 additions & 1 deletion frontend/components/YourWalletPage/WithdrawFunds.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ import React, { useCallback, useMemo, useState } from 'react';

import { COLOR } from '@/constants/colors';
import { useBalanceContext } from '@/hooks/useBalanceContext';
import { useFeatureFlag } from '@/hooks/useFeatureFlag';
import { useService } from '@/hooks/useService';
import { useServices } from '@/hooks/useServices';
import { useStakingContractCountdown } from '@/hooks/useStakingContractCountdown';
Expand All @@ -14,6 +15,7 @@ import { ServicesService } from '@/service/Services';
import { Address } from '@/types/Address';

import { CustomAlert } from '../Alert';
import { FeatureNotEnabled } from '../FeatureNotEnabled';

const { Text } = Typography;

Expand Down Expand Up @@ -54,7 +56,6 @@ const CompatibleMessage = () => (
export const WithdrawFunds = () => {
const { selectedService, refetch: refetchServices } = useServices();
const { refetch: refetchMasterWallets } = useMasterWalletContext();

const { updateBalances } = useBalanceContext();

const { service, isServiceRunning } = useService(
Expand All @@ -64,6 +65,7 @@ export const WithdrawFunds = () => {
const { isServiceStakedForMinimumDuration, selectedStakingContractDetails } =
useActiveStakingContractDetails();

// state
const [isModalVisible, setIsModalVisible] = useState(false);
const [withdrawAddress, setWithdrawAddress] = useState('');
const [isWithdrawalLoading, setIsWithdrawalLoading] = useState(false);
Expand Down Expand Up @@ -148,6 +150,9 @@ export const WithdrawFunds = () => {
[service, isServiceStakedForMinimumDuration, showModal],
);

const isWithdrawFundsEnabled = useFeatureFlag('withdraw-funds');
if (!isWithdrawFundsEnabled) return <FeatureNotEnabled />;

return (
<>
{isServiceStakedForMinimumDuration ? (
Expand Down
134 changes: 80 additions & 54 deletions frontend/components/YourWalletPage/YourAgent.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,16 +4,17 @@ import Image from 'next/image';
import { useMemo } from 'react';
import styled from 'styled-components';

import { MiddlewareChain } from '@/client';
import { OLAS_CONTRACTS } from '@/config/olasContracts';
import { UNICODE_SYMBOLS } from '@/constants/symbols';
import { EvmChainId } from '@/enums/Chain';
import { NA, UNICODE_SYMBOLS } from '@/constants/symbols';
import { BLOCKSCOUT_URL_BY_MIDDLEWARE_CHAIN } from '@/constants/urls';
import { ContractType } from '@/enums/Contract';
import { TokenSymbol } from '@/enums/Token';
import { AgentSafe, Safe } from '@/enums/Wallet';
import {
useBalanceContext,
useServiceBalances,
} from '@/hooks/useBalanceContext';
import { useFeatureFlag } from '@/hooks/useFeatureFlag';
import { useReward } from '@/hooks/useReward';
import { useService } from '@/hooks/useService';
import { useServices } from '@/hooks/useServices';
Expand All @@ -26,6 +27,7 @@ import { AddressLink } from '../AddressLink';
import { InfoBreakdownList } from '../InfoBreakdown';
import { Container, infoBreakdownParentStyle } from './styles';
import { OlasTitle, OwnershipNftTitle, ServiceNftIdTitle } from './Titles';
import { useYourWallet } from './useYourWallet';
import { WithdrawFunds } from './WithdrawFunds';

const { Text, Paragraph } = Typography;
Expand All @@ -39,8 +41,8 @@ const NftCard = styled(Card)`
}
`;

const SafeAddress = ({ serviceSafe }: { serviceSafe: Safe }) => {
const multisigAddress = serviceSafe.address;
const SafeAddress = ({ address }: { address: Address }) => {
const { middlewareChain } = useYourWallet();

return (
<Flex vertical gap={8}>
Expand All @@ -49,7 +51,12 @@ const SafeAddress = ({ serviceSafe }: { serviceSafe: Safe }) => {
{
left: 'Wallet Address',
leftClassName: 'text-light text-sm',
right: <AddressLink address={multisigAddress} />,
right: (
<AddressLink
address={address}
middlewareChain={middlewareChain}
/>
),
rightClassName: 'font-normal text-sm',
},
]}
Expand All @@ -59,11 +66,16 @@ const SafeAddress = ({ serviceSafe }: { serviceSafe: Safe }) => {
);
};

const AgentTitle = ({ serviceSafe }: { serviceSafe: AgentSafe }) => {
const agentName = useMemo(
() => (serviceSafe ? generateName(serviceSafe.address) : '--'),
[serviceSafe],
);
const AgentTitle = ({ address }: { address: Address }) => {
const { middlewareChain } = useYourWallet();

const agentProfileLink = useMemo(() => {
if (!address) return null;
if (middlewareChain === MiddlewareChain.GNOSIS) {
return `https://predict.olas.network/agents/${address}`;
}
return null;
}, [address, middlewareChain]);

return (
<Flex vertical gap={12}>
Expand All @@ -89,30 +101,29 @@ const AgentTitle = ({ serviceSafe }: { serviceSafe: AgentSafe }) => {
}
placement="top"
>
<Text strong>{agentName}</Text>
<Text strong>{address ? generateName(address) : NA}</Text>
</Tooltip>
{/* TODO: address multi-agent at later point */}
<a
href={`https://predict.olas.network/agents/${serviceSafe.address}`}
target="_blank"
className="text-sm"
>
Agent profile {UNICODE_SYMBOLS.EXTERNAL_LINK}
</a>

{agentProfileLink && (
<a href={agentProfileLink} target="_blank" className="text-sm">
Agent profile {UNICODE_SYMBOLS.EXTERNAL_LINK}
</a>
)}
</Flex>
</Flex>
</Flex>
</Flex>
);
};

type ServiceAndNftDetailsProps = { serviceNftTokenId: number };
const ServiceAndNftDetails = ({
serviceNftTokenId,
}: {
serviceNftTokenId: number;
}) => {
}: ServiceAndNftDetailsProps) => {
const { middlewareChain, evmHomeChainId } = useYourWallet();

const serviceRegistryL2ContractAddress =
OLAS_CONTRACTS[EvmChainId.Gnosis][ContractType.ServiceRegistryL2].address;
OLAS_CONTRACTS[evmHomeChainId][ContractType.ServiceRegistryL2].address;

return (
<NftCard>
Expand All @@ -129,7 +140,7 @@ const ServiceAndNftDetails = ({
<Flex vertical>
<OwnershipNftTitle />
<a
href={`https://gnosis.blockscout.com/token/${serviceRegistryL2ContractAddress}/instance/${serviceNftTokenId}`}
href={`${BLOCKSCOUT_URL_BY_MIDDLEWARE_CHAIN[middlewareChain]}/token/${serviceRegistryL2ContractAddress}/instance/${serviceNftTokenId}`}
target="_blank"
>
{truncateAddress(serviceRegistryL2ContractAddress as Address)}{' '}
Expand All @@ -140,7 +151,7 @@ const ServiceAndNftDetails = ({
<Flex vertical>
<ServiceNftIdTitle />
<a
href={`https://registry.olas.network/gnosis/services/${serviceNftTokenId}`}
href={`https://registry.olas.network/${middlewareChain}/services/${serviceNftTokenId}`}
target="_blank"
>
{serviceNftTokenId} {UNICODE_SYMBOLS.EXTERNAL_LINK}
Expand All @@ -155,12 +166,13 @@ const ServiceAndNftDetails = ({
const YourAgentWalletBreakdown = () => {
const { isLoaded } = useBalanceContext();
const { selectedService } = useServices();
const { serviceSafes, serviceNftTokenId, serviceEoa } = useService(
const { serviceNftTokenId, serviceEoa } = useService(
selectedService?.service_config_id,
);
const { serviceSafeBalances, serviceEoaBalances } = useServiceBalances(
selectedService?.service_config_id,
);
const { serviceSafe, middlewareChain, evmHomeChainId } = useYourWallet();

const {
availableRewardsForEpochEth,
Expand All @@ -170,23 +182,27 @@ const YourAgentWalletBreakdown = () => {

const reward = useMemo(() => {
if (!isLoaded) return <Skeleton.Input size="small" active />;
if (!isEligibleForRewards) return 'Not yet earned';
return `~${balanceFormat(availableRewardsForEpochEth, 2)} OLAS`;
if (isEligibleForRewards) {
return `~${balanceFormat(availableRewardsForEpochEth, 2)} OLAS`;
}

return 'Not yet earned';
}, [isLoaded, isEligibleForRewards, availableRewardsForEpochEth]);

const serviceSafeOlasBalances = useMemo(
const serviceSafeOlas = useMemo(
() =>
serviceSafeBalances?.filter(
(walletBalance) => walletBalance.symbol === TokenSymbol.OLAS,
serviceSafeBalances?.find(
({ symbol, evmChainId }) =>
symbol === TokenSymbol.OLAS && evmChainId === evmHomeChainId,
),
[serviceSafeBalances],
[serviceSafeBalances, evmHomeChainId],
);

const serviceSafeRewards = useMemo(
() => [
{
title: 'Claimed rewards',
value: `${balanceFormat(serviceSafeOlasBalances?.[0]?.balance ?? 0, 2)} OLAS`,
value: `${balanceFormat(serviceSafeOlas?.balance ?? 0, 2)} OLAS`,
},
{
title: 'Unclaimed rewards',
Expand All @@ -197,30 +213,31 @@ const YourAgentWalletBreakdown = () => {
value: reward,
},
],
[accruedServiceStakingRewards, reward, serviceSafeOlasBalances],
[accruedServiceStakingRewards, reward, serviceSafeOlas],
);

const serviceSafeNativeBalances = useMemo(
() => serviceSafeBalances?.filter((balance) => balance.isNative),
[serviceSafeBalances],
() =>
serviceSafeBalances?.filter(
({ isNative, evmChainId }) => isNative && evmChainId === evmHomeChainId,
),
[serviceSafeBalances, evmHomeChainId],
);

const serviceEoaNativeBalances = useMemo(
() => serviceEoaBalances?.filter((balance) => balance.isNative),
[serviceEoaBalances],
() =>
serviceEoaBalances?.filter(
({ isNative, evmChainId }) => isNative && evmChainId === evmHomeChainId,
),
[serviceEoaBalances, evmHomeChainId],
);

const serviceSafe = useMemo(() => {
if (isNil(serviceSafes) || isEmpty(serviceSafes)) return null;
return serviceSafes[0];
}, [serviceSafes]);

if (isNil(serviceSafe)) return null;
if (!serviceSafe) return null;

return (
<Card title={<AgentTitle serviceSafe={serviceSafe} />}>
<Card title={<AgentTitle address={serviceSafe.address} />}>
<Container>
<SafeAddress serviceSafe={serviceSafe} />
<SafeAddress address={serviceSafe.address} />

{!isEmpty(serviceSafeRewards) && (
<Flex vertical gap={8}>
Expand Down Expand Up @@ -255,7 +272,12 @@ const YourAgentWalletBreakdown = () => {
{
left: 'Signer',
leftClassName: 'text-sm',
right: <AddressLink address={serviceEoa.address} />,
right: (
<AddressLink
address={serviceEoa.address}
middlewareChain={middlewareChain}
/>
),
rightClassName: 'font-normal text-sm',
},
]}
Expand All @@ -273,9 +295,13 @@ const YourAgentWalletBreakdown = () => {
);
};

export const YourAgentWallet = () => (
<>
<YourAgentWalletBreakdown />
<WithdrawFunds />
</>
);
export const YourAgentWallet = () => {
const isWithdrawFundsEnabled = useFeatureFlag('withdraw-funds');

return (
<>
<YourAgentWalletBreakdown />
{isWithdrawFundsEnabled && <WithdrawFunds />}
</>
);
};
Loading

0 comments on commit 7c2ba1c

Please sign in to comment.