Skip to content

Commit

Permalink
feat: added bitcoin contract list page and entry point
Browse files Browse the repository at this point in the history
  • Loading branch information
Polybius93 committed Oct 18, 2023
1 parent ce13789 commit a28678d
Show file tree
Hide file tree
Showing 13 changed files with 416 additions and 25 deletions.
100 changes: 82 additions & 18 deletions src/app/common/hooks/use-bitcoin-contracts.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,32 +4,32 @@ import { RpcErrorCode } from '@btckit/types';
import { JsDLCInterface } from '@dlc-link/dlc-tools';
import { bytesToHex } from '@stacks/common';

import { BITCOIN_API_BASE_URL_MAINNET, BITCOIN_API_BASE_URL_TESTNET } from '@shared/constants';
import {
deriveAddressIndexKeychainFromAccount,
extractAddressIndexFromPath,
} from '@shared/crypto/bitcoin/bitcoin.utils';
import { createMoneyFromDecimal } from '@shared/models/money.model';
import { Money, createMoneyFromDecimal } from '@shared/models/money.model';
import { RouteUrls } from '@shared/route-urls';
import { BitcoinContractResponseStatus } from '@shared/rpc/methods/accept-bitcoin-contract';
import { makeRpcSuccessResponse } from '@shared/rpc/rpc-methods';
import { makeRpcErrorResponse } from '@shared/rpc/rpc-methods';

import { checkDlcLinkAttestorHealth } from '@app/query/bitcoin/contract/check-dlc-link-attestor-health';
import { sendAcceptedBitcoinContractOfferToProtocolWallet } from '@app/query/bitcoin/contract/send-accepted-bitcoin-contract-offer';
import {
useCalculateBitcoinFiatValue,
useCryptoCurrencyMarketData,
} from '@app/query/common/market-data/market-data.hooks';
import { useCurrentAccountIndex } from '@app/store/accounts/account';
import {
useCurrentAccountNativeSegwitSigner,
useCurrentAccountNativeSegwitIndexZeroSigner,
useNativeSegwitAccountBuilder,
} from '@app/store/accounts/blockchain/bitcoin/native-segwit-account.hooks';
import { useCurrentNetwork } from '@app/store/networks/networks.selectors';

import { initialSearchParams } from '../initial-search-params';
import { i18nFormatCurrency } from '../money/format-money';
import { satToBtc } from '../money/unit-conversion';
import { whenBitcoinNetwork } from '../utils';
import { useDefaultRequestParams } from './use-default-request-search-params';

export interface SimplifiedBitcoinContract {
Expand All @@ -44,6 +44,13 @@ interface CounterpartyWalletDetails {
counterpartyWalletIcon: string;
}

export interface BitcoinContractListItem {
id: string;
state: string;
acceptorCollateral: string;
txId: string;
}

export interface BitcoinContractOfferDetails {
simplifiedBitcoinContract: SimplifiedBitcoinContract;
counterpartyWalletDetails: CounterpartyWalletDetails;
Expand All @@ -54,18 +61,16 @@ export function useBitcoinContracts() {
const defaultParams = useDefaultRequestParams();
const bitcoinMarketData = useCryptoCurrencyMarketData('BTC');
const calculateFiatValue = useCalculateBitcoinFiatValue();
const getNativeSegwitSigner = useCurrentAccountNativeSegwitSigner();
const bitcoinAccountDetails = useCurrentAccountNativeSegwitIndexZeroSigner();
const currentIndex = useCurrentAccountIndex();
const nativeSegwitPrivateKeychain = useNativeSegwitAccountBuilder()?.(currentIndex);
const currentBitcoinNetwork = useCurrentNetwork();

async function getBitcoinContractInterface(
attestorURLs: string[]
): Promise<JsDLCInterface | undefined> {
const bitcoinAccountDetails = getNativeSegwitSigner?.(0);

if (!nativeSegwitPrivateKeychain || !bitcoinAccountDetails) return;

const currentBitcoinNetwork = bitcoinAccountDetails.network;
const currentAddress = bitcoinAccountDetails.address;
const currentAccountIndex = extractAddressIndexFromPath(bitcoinAccountDetails.derivationPath);

Expand All @@ -75,18 +80,11 @@ export function useBitcoinContracts() {

if (!currentAddressPrivateKey) return;

const blockchainAPI = whenBitcoinNetwork(currentBitcoinNetwork)({
mainnet: BITCOIN_API_BASE_URL_MAINNET,
testnet: BITCOIN_API_BASE_URL_TESTNET,
regtest: BITCOIN_API_BASE_URL_TESTNET,
signet: BITCOIN_API_BASE_URL_TESTNET,
});

const bitcoinContractInterface = JsDLCInterface.new(
const bitcoinContractInterface = await JsDLCInterface.new(
bytesToHex(currentAddressPrivateKey),
currentAddress,
currentBitcoinNetwork,
blockchainAPI,
currentBitcoinNetwork.chain.bitcoin.network,
currentBitcoinNetwork.chain.bitcoin.url,
JSON.stringify(attestorURLs)
);

Expand Down Expand Up @@ -198,6 +196,57 @@ export function useBitcoinContracts() {
close();
}

async function getHealthyDlcLinkAttestor(): Promise<string> {
const dlcLinkAttestorUrls = [
'https://devnet.dlc.link/attestor-1/',
'https://devnet.dlc.link/attestor-2/',
'https://devnet.dlc.link/attestor-3/',
];

let currentAttestorUrl: string | undefined;

for (const attestorURL of dlcLinkAttestorUrls) {
const isAttestorHealthy = await checkDlcLinkAttestorHealth(attestorURL);
if (isAttestorHealthy) {
currentAttestorUrl = attestorURL;
break;
}
}

if (!currentAttestorUrl) {
throw new Error('Unable to find a healthy DLC.Link attestor');
}

return currentAttestorUrl;
}

async function getAllSignedBitcoinContracts() {
let bitcoinContractInterface: JsDLCInterface | undefined;

try {
const currentAttestorUrl = await getHealthyDlcLinkAttestor();
bitcoinContractInterface = await getBitcoinContractInterface([currentAttestorUrl]);
} catch (error) {
navigate(RouteUrls.BitcoinContractLockError, {
state: {
error,
title: 'There was an error with getting the Bitcoin Contract Interface',
body: 'Unable to setup Bitcoin Contract Interface',
},
});
sendRpcResponse(BitcoinContractResponseStatus.INTERFACE_ERROR);
}

if (!bitcoinContractInterface) return;

const bitcoinContracts = await bitcoinContractInterface.get_contracts();
const signedBitcoinContracts = bitcoinContracts.filter(
(bitcoinContract: BitcoinContractListItem) => bitcoinContract.state === 'Signed'
);

return signedBitcoinContracts;
}

function getTransactionDetails(txId: string, bitcoinCollateral: number) {
const bitcoinValue = satToBtc(bitcoinCollateral);
const txMoney = createMoneyFromDecimal(bitcoinValue, 'BTC');
Expand All @@ -215,6 +264,19 @@ export function useBitcoinContracts() {
};
}

async function sumBitcoinContractCollateralAmounts(): Promise<Money> {
let bitcoinContractsCollateralSum = 0;
const bitcoinContracts = await getAllSignedBitcoinContracts();
bitcoinContracts.forEach((bitcoinContract: BitcoinContractListItem) => {
bitcoinContractsCollateralSum += parseInt(bitcoinContract.acceptorCollateral);
});
const bitcoinContractCollateralSumMoney = createMoneyFromDecimal(
satToBtc(bitcoinContractsCollateralSum),
'BTC'
);
return bitcoinContractCollateralSumMoney;
}

function sendRpcResponse(
responseStatus: BitcoinContractResponseStatus,
bitcoinContractId?: string,
Expand Down Expand Up @@ -275,6 +337,8 @@ export function useBitcoinContracts() {
handleOffer,
handleAccept,
handleReject,
getAllSignedBitcoinContracts,
sumBitcoinContractCollateralAmounts,
sendRpcResponse,
};
}
4 changes: 0 additions & 4 deletions src/app/common/utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -264,10 +264,6 @@ export function whenStacksChainId(chainId: ChainID) {
return <T>(chainIdMap: WhenStacksChainIdMap<T>): T => chainIdMap[chainId];
}

export function whenBitcoinNetwork(mode: BitcoinNetworkModes) {
return <T>(networkMap: Record<BitcoinNetworkModes, T>): T => networkMap[mode];
}

export function logAndThrow(msg: string, args: any[] = []) {
logger.error(msg, ...args);
throw new Error(msg);
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
import { Flex, StackProps } from '@stacks/ui';
import { forwardRefWithAs } from '@stacks/ui-core';
import { CryptoAssetSelectors } from '@tests/selectors/crypto-asset.selectors';
import { HStack } from 'leather-styles/jsx';

import { Money } from '@shared/models/money.model';

import { formatBalance } from '@app/common/format-balance';
import { ftDecimals } from '@app/common/stacks-utils';
import { Flag } from '@app/components/layout/flag';
import { Tooltip } from '@app/components/tooltip';
import { Caption, Text } from '@app/components/typography';

import { SmallLoadingSpinner } from '../loading-spinner';

interface BitcoinContractEntryPointLayoutProps extends StackProps {
balance: Money;
caption: string;
icon: React.JSX.Element;
usdBalance?: string;
isLoading?: boolean;
onClick: () => void;
}
export const BitcoinContractEntryPointLayout = forwardRefWithAs(
(props: BitcoinContractEntryPointLayoutProps) => {
const { balance, caption, icon, usdBalance, isLoading, onClick } = props;

const amount = balance.decimals
? ftDecimals(balance.amount, balance.decimals)
: balance.amount.toString();
const dataTestId = CryptoAssetSelectors.CryptoAssetListItem.replace(
'{symbol}',
balance.symbol.toLowerCase()
);
const formattedBalance = formatBalance(amount);

return (
<Flex as={'button'} onClick={onClick} data-testid={dataTestId} outline={0}>
<Flag img={icon} align="middle" spacing="base" width="100%">
<HStack alignItems="center" justifyContent="space-between" width="100%">
<Text>{'Bitcoin Contracts'}</Text>
<Tooltip
label={formattedBalance.isAbbreviated ? balance.amount.toString() : undefined}
placement="left-start"
>
<Text
data-testid={'Bitcoin Contracts'}
fontVariantNumeric="tabular-nums"
textAlign="right"
>
{isLoading ? <SmallLoadingSpinner /> : formattedBalance.value}
</Text>
</Tooltip>
</HStack>
<HStack height="1.25rem" alignItems="center" justifyContent="space-between" width="100%">
<Caption>{caption}</Caption>
<Flex>{isLoading ? '' : <Caption>{usdBalance}</Caption>}</Flex>
</HStack>
</Flag>
</Flex>
);
}
);
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
import { useEffect, useState } from 'react';
import { useNavigate } from 'react-router-dom';

import { Box } from '@stacks/ui';

import { Money, createMoneyFromDecimal } from '@shared/models/money.model';
import { RouteUrls } from '@shared/route-urls';

import { useBitcoinContracts } from '@app/common/hooks/use-bitcoin-contracts';
import { i18nFormatCurrency } from '@app/common/money/format-money';
import { useCalculateBitcoinFiatValue } from '@app/query/common/market-data/market-data.hooks';

import { BitcoinContractIcon } from '../icons/bitcoin-contract-icon';
import { BitcoinContractEntryPointLayout } from './bitcoin-contract-entry-point-layout';

interface BitcoinContractEntryPointProps {
btcAddress: string;
}

export function BitcoinContractEntryPoint({ btcAddress }: BitcoinContractEntryPointProps) {
const navigate = useNavigate();
const { sumBitcoinContractCollateralAmounts } = useBitcoinContracts();
const [isLoading, setIsLoading] = useState(true);
const calculateFiatValue = useCalculateBitcoinFiatValue();
const [bitcoinContractSum, setBitcoinContractSum] = useState<Money>(
createMoneyFromDecimal(0, 'BTC')
);

useEffect(() => {
const getBitcoinContractDataAndSetState = async () => {
setIsLoading(true);
const currentBitcoinContractSum = await sumBitcoinContractCollateralAmounts();
setBitcoinContractSum(currentBitcoinContractSum);
setIsLoading(false);
};
getBitcoinContractDataAndSetState();
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [btcAddress]);

function onClick() {
navigate(RouteUrls.BitcoinContractList);
}

return (
<BitcoinContractEntryPointLayout
isLoading={isLoading}
cursor={'pointer'}
balance={bitcoinContractSum}
caption={bitcoinContractSum.symbol}
icon={<Box as={BitcoinContractIcon} />}
usdBalance={i18nFormatCurrency(calculateFiatValue(bitcoinContractSum))}
onClick={onClick}
/>
);
}
40 changes: 40 additions & 0 deletions src/app/components/icons/bitcoin-contract-icon.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
export function BitcoinContractIcon() {
return (
<svg width="36" height="36" viewBox="0 0 36 36" fill="none" xmlns="http://www.w3.org/2000/svg">
<g clipPath="url(#clip0_7_6)">
<path
d="M18 36C27.9411 36 36 27.9411 36 18C36 8.05887 27.9411 0 18 0C8.05887 0 0 8.05887 0 18C0 27.9411 8.05887 36 18 36Z"
fill="url(#paint0_radial_7_6)"
/>
<path
fillRule="evenodd"
clipRule="evenodd"
d="M14.5527 10.9136C14.5527 9.77371 14.8658 8.93453 15.373 8.38673C15.8725 7.84728 16.6434 7.5 17.7519 7.5C18.861 7.5 19.6308 7.84662 20.129 8.38479C20.635 8.93138 20.9474 9.76945 20.9474 10.91V13.114H14.5527V10.9136ZM13.0527 13.114V10.9136C13.0527 9.50789 13.4413 8.26526 14.2723 7.36767C15.1111 6.46172 16.3148 6 17.7519 6C19.1886 6 20.3916 6.46045 21.2297 7.3658C22.06 8.26274 22.4474 9.50467 22.4474 10.91V13.114H24.4071C25.127 13.114 25.7105 13.6976 25.7105 14.4174V16.4298H24.2105V14.614H11.5V23.3772H17.8553L17.4307 24.9307L11.3034 24.8772C10.5836 24.8772 10 24.2936 10 23.5737V14.4174C10 13.6976 10.5836 13.114 11.3034 13.114H13.0527Z"
fill="white"
/>
<path
fillRule="evenodd"
clipRule="evenodd"
d="M23.3811 19.1028C24.703 19.2333 25.7395 19.6373 25.8544 20.8407C25.9324 21.7181 25.5625 22.2401 24.9509 22.5343C25.9384 22.7807 26.539 23.3602 26.4149 24.6457C26.2638 26.2569 25.0584 26.6753 23.3351 26.7636L23.3379 28.427L22.3229 28.4284L22.3202 26.7649C22.2093 26.7633 22.0934 26.765 21.9745 26.7667H21.9744C21.8155 26.769 21.6512 26.7713 21.4866 26.766L21.4893 28.4294L20.4763 28.4307L20.4736 26.7673L18.4439 26.7272L18.615 25.5115C18.615 25.5115 19.3815 25.5284 19.3673 25.5196C19.6495 25.5181 19.7333 25.318 19.7541 25.1866L19.7491 22.5201L19.7571 20.6223C19.7259 20.4245 19.5913 20.1871 19.1764 20.1799C19.199 20.1597 18.4333 20.1809 18.4333 20.1809L18.4307 19.0664L20.5106 19.0638L20.508 17.4344L21.5558 17.4331L21.5584 19.0624C21.7607 19.0568 21.9602 19.0584 22.162 19.06C22.2283 19.0606 22.2947 19.0611 22.3616 19.0614L22.359 17.432L23.3784 17.4307L23.3811 19.1028ZM21.5707 22.2764L21.615 22.2784C22.1453 22.3015 23.7176 22.37 23.72 21.4167C23.7176 20.5117 22.3794 20.5409 21.7577 20.5545C21.6842 20.5561 21.6208 20.5574 21.5707 20.557V22.2764ZM21.6451 25.0843C21.6173 25.0832 21.5916 25.0821 21.5683 25.0812L21.5683 23.3618C21.6237 23.3623 21.6934 23.3615 21.7739 23.3605C22.5087 23.3513 24.15 23.3309 24.1422 24.2433C24.1505 25.1871 22.2979 25.1111 21.6451 25.0843Z"
fill="white"
/>
</g>
<defs>
<radialGradient
id="paint0_radial_7_6"
cx="0"
cy="0"
r="1"
gradientUnits="userSpaceOnUse"
gradientTransform="translate(6.9471 1.64) scale(37.7884)"
>
<stop offset="0.1011" stopColor="#93009E" />
<stop offset="1" stopColor="#001FBA" />
</radialGradient>
<clipPath id="clip0_7_6">
<rect width="36" height="36" fill="white" />
</clipPath>
</defs>
</svg>
);
}
8 changes: 8 additions & 0 deletions src/app/components/loading-spinner.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,14 @@ export function LoadingSpinner(props: FlexProps) {
);
}

export function SmallLoadingSpinner(props: FlexProps) {
return (
<Flex alignItems="center" flexGrow={1} justifyContent="center" width="100%" {...props}>
<Spinner color={color('text-caption')} opacity={0.5} size="sm" />
</Flex>
);
}

export function FullPageLoadingSpinner(props: FlexProps) {
return (
<Flex height="100vh" width="100%" {...props}>
Expand Down
Loading

0 comments on commit a28678d

Please sign in to comment.