Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Stage -> Main #1118

Merged
merged 4 commits into from
Nov 18, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion apps/dapp/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -52,7 +52,8 @@
"tippy.js": "^6.3.7",
"use-debounce": "^9.0.2",
"use-interval": "^1.4.0",
"util": "^0.12.4"
"util": "^0.12.4",
"zod": "^3.19.1"
},
"devDependencies": {
"@babel/core": "^7.16.0",
Expand Down
21 changes: 11 additions & 10 deletions apps/dapp/src/components/Pages/Core/DappPages/Borrow/Chart.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@ import {
formatNumberFixedDecimals,
} from 'utils/formatter';
import { formatDailyDataPoints } from 'utils/charts';
import { fetchGenericSubgraph } from 'utils/subgraph';
import { queryTlcDailySnapshots, subgraphQuery } from 'utils/subgraph';
import IntervalToggler from 'components/Charts/IntervalToggler';
import env from 'constants/env';

Expand Down Expand Up @@ -50,17 +50,18 @@ export const TlcChart = () => {

useEffect(() => {
const fetchMetrics = async () => {
const { data } = await fetchGenericSubgraph<any>(
const response = await subgraphQuery(
env.subgraph.templeV2,
`{
tlcDailySnapshots(orderBy: timestamp, orderDirection: desc) {
timestamp
utilRatio
interestYield
}
}`
queryTlcDailySnapshots()
);

setMetrics(
response.tlcDailySnapshots.map((r) => ({
timestamp: parseFloat(r.timestamp),
utilRatio: parseFloat(r.utilRatio),
interestYield: parseFloat(r.interestYield),
}))
);
setMetrics(data.tlcDailySnapshots);
};
fetchMetrics();
}, []);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ import {
TlcInfo,
Warning,
} from '../index';
import { fromAtto, toAtto } from 'utils/bigNumber';
import { fromAtto, toAtto, ZERO } from 'utils/bigNumber';
import styled from 'styled-components';
import { ReactNode, useEffect, useMemo, useState } from 'react';

Expand Down Expand Up @@ -67,18 +67,28 @@ export const Borrow: React.FC<IProps> = ({

const userMaxBorrowBigNumber = toAtto(userMaxBorrow);

if (!tlcInfo) {
return { value: userMaxBorrow, isCircuitBreakerActive: false };
}
let returnValue = { value: userMaxBorrow, isCircuitBreakerActive: false };

if (tlcInfo.daiCircuitBreakerRemaining.lt(userMaxBorrowBigNumber)) {
return {
// Check if the dai circuit breaker is active
if (
tlcInfo &&
tlcInfo.daiCircuitBreakerRemaining.lt(userMaxBorrowBigNumber)
) {
returnValue = {
value: fromAtto(tlcInfo.daiCircuitBreakerRemaining),
isCircuitBreakerActive: true,
};
}

return { value: userMaxBorrow, isCircuitBreakerActive: false };
// Check if trvAvailable from the contract is less than the user max borrow
if (tlcInfo && tlcInfo.trvAvailable.lt(userMaxBorrowBigNumber)) {
returnValue = {
value: fromAtto(tlcInfo.trvAvailable || ZERO),
isCircuitBreakerActive: true,
};
}

return returnValue;
}, [tlcInfo, accountPosition, prices.tpi]);

return (
Expand Down
81 changes: 55 additions & 26 deletions apps/dapp/src/components/Pages/Core/DappPages/Borrow/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,11 @@ import {
TreasuryReservesVault__factory,
} from 'types/typechain';
import { ITlcDataTypes } from 'types/typechain/contracts/interfaces/v2/templeLineOfCredit/ITempleLineOfCredit';
import { fetchGenericSubgraph } from 'utils/subgraph';
import {
queryTlcMinBorrowAmount,
queryTlcPrices,
subgraphQuery,
} from 'utils/subgraph';
import { BigNumber, ethers } from 'ethers';
import daiImg from 'assets/images/newui-images/tokens/dai.png';
import templeImg from 'assets/images/newui-images/tokens/temple.png';
Expand Down Expand Up @@ -48,6 +52,8 @@ export type TlcInfo = {
debtCeiling: number;
daiCircuitBreakerRemaining: BigNumber;
templeCircuitBreakerRemaining: BigNumber;
outstandingUserDebt: number;
trvAvailable: BigNumber;
};

export const MAX_LTV = 85;
Expand Down Expand Up @@ -86,23 +92,18 @@ export const BorrowPage = () => {
const [metricsLoading, setMetricsLoading] = useState(false);

const getPrices = useCallback(async () => {
const { data } = await fetchGenericSubgraph<any>(
const response = await subgraphQuery(
env.subgraph.templeV2,
`{
tokens {
price
symbol
}
treasuryReservesVaults {
treasuryPriceIndex
}
}`
queryTlcPrices()
);
setPrices({
templePrice: data.tokens.filter((t: any) => t.symbol == 'TEMPLE')[0]
.price,
daiPrice: data.tokens.filter((t: any) => t.symbol == 'DAI')[0].price,
tpi: Number(data.treasuryReservesVaults[0].treasuryPriceIndex),
templePrice: parseFloat(
response.tokens.filter((t: any) => t.symbol == 'TEMPLE')[0].price
),
daiPrice: parseFloat(
response.tokens.filter((t: any) => t.symbol == 'DAI')[0].price
),
tpi: parseFloat(response.treasuryReservesVaults[0].treasuryPriceIndex),
});
}, []);

Expand Down Expand Up @@ -148,6 +149,7 @@ export const BorrowPage = () => {
const debtPosition = await tlcContract.totalDebtPosition();
const totalUserDebt = debtPosition.totalDebt;
const utilizationRatio = debtPosition.utilizationRatio;
const outstandingUserDebt = debtPosition[2];

// NOTE: We are intentionally rounding here to nearest 1e18
const debtCeiling = totalUserDebt
Expand All @@ -159,6 +161,9 @@ export const BorrowPage = () => {
const trvContract = new TreasuryReservesVault__factory(signer).attach(
env.contracts.treasuryReservesVault
);

const trvAvailable = await trvContract.totalAvailable(env.contracts.dai);

const strategyAvailalableToBorrowFromTrv =
await trvContract.availableForStrategyToBorrow(
env.contracts.strategies.tlcStrategy,
Expand Down Expand Up @@ -189,6 +194,8 @@ export const BorrowPage = () => {
strategyBalance: fromAtto(maxAvailableToBorrow),
borrowRate: currentBorrowInterestRate,
liquidationLtv: fromAtto(maxLtv),
outstandingUserDebt: fromAtto(outstandingUserDebt),
trvAvailable: trvAvailable,
daiCircuitBreakerRemaining: circuitBreakers?.daiCircuitBreakerRemaining,
templeCircuitBreakerRemaining:
circuitBreakers?.templeCircuitBreakerRemaining,
Expand All @@ -210,13 +217,9 @@ export const BorrowPage = () => {
};
getAccountPosition();
try {
const { data } = await fetchGenericSubgraph<any>(
const response = await subgraphQuery(
env.subgraph.templeV2,
`{
tlcDailySnapshots(orderBy: timestamp, orderDirection: desc, first: 1) {
minBorrowAmount
}
}`
queryTlcMinBorrowAmount()
);

const tlcInfoFromContracts = await getTlcInfoFromContracts();
Expand All @@ -230,7 +233,7 @@ export const BorrowPage = () => {
}

setTlcInfo({
minBorrow: data.tlcDailySnapshots[0].minBorrowAmount,
minBorrow: parseFloat(response.tlcDailySnapshots[0].minBorrowAmount),
borrowRate: tlcInfoFromContracts?.borrowRate || 0,
liquidationLtv: tlcInfoFromContracts?.liquidationLtv || 0,
strategyBalance: tlcInfoFromContracts?.strategyBalance || 0,
Expand All @@ -239,6 +242,8 @@ export const BorrowPage = () => {
tlcInfoFromContracts?.daiCircuitBreakerRemaining || ZERO,
templeCircuitBreakerRemaining:
tlcInfoFromContracts?.templeCircuitBreakerRemaining || ZERO,
outstandingUserDebt: tlcInfoFromContracts?.outstandingUserDebt || 0,
trvAvailable: tlcInfoFromContracts?.trvAvailable || ZERO,
});
} catch (e) {
setMetricsLoading(false);
Expand Down Expand Up @@ -482,14 +487,21 @@ export const BorrowPage = () => {
if (!tlcInfo) return '...';

const availableAsBigNumber = toAtto(tlcInfo.strategyBalance);
let borrowableAmount = tlcInfo.strategyBalance;

if (tlcInfo.daiCircuitBreakerRemaining.lt(availableAsBigNumber)) {
return `$${Number(
fromAtto(tlcInfo.daiCircuitBreakerRemaining)
).toLocaleString()}`;
borrowableAmount = fromAtto(tlcInfo.daiCircuitBreakerRemaining);
}

const trvAvailable = fromAtto(tlcInfo.trvAvailable);
if (trvAvailable < borrowableAmount) {
borrowableAmount = trvAvailable;
}

return `$${Number(tlcInfo.strategyBalance).toLocaleString()}`;
return `$${Number(borrowableAmount).toLocaleString('en', {
minimumFractionDigits: 2,
maximumFractionDigits: 2,
})}`;
}, [tlcInfo]);

return (
Expand Down Expand Up @@ -717,12 +729,29 @@ export const BorrowPage = () => {
</LeadMetric>
<BrandParagraph>Current Borrow APY </BrandParagraph>
</MetricContainer>
</Metrics>
<Metrics>
<MetricContainer>
<LeadMetric>
{showLoading ? '...' : prices.tpi.toFixed(2)}
</LeadMetric>
<BrandParagraph>Current TPI</BrandParagraph>
</MetricContainer>
<MetricContainer>
<LeadMetric>
{showLoading
? '...'
: tlcInfo &&
`$${Number(tlcInfo.outstandingUserDebt).toLocaleString(
'en',
{
minimumFractionDigits: 2,
maximumFractionDigits: 2,
}
)}`}
</LeadMetric>
<BrandParagraph>Outstanding User Debt</BrandParagraph>
</MetricContainer>
</Metrics>
<ChartContainer>
<TlcChart />
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,13 +7,14 @@ import { formatTimestampedChartData } from 'utils/charts';
import useV2StrategySnapshotData, {
StrategyTokenField,
V2SnapshotMetric,
V2StrategySnapshot,
} from '../hooks/use-dashboardv2-daily-snapshots';
import {
ALL_STRATEGIES,
DashboardData,
isTRVDashboard,
StrategyKey,
} from '../DashboardConfig';
import { V2StrategySnapshot } from 'utils/subgraph';

type XAxisTickFormatter = (timestamp: number) => string;

Expand Down Expand Up @@ -181,12 +182,16 @@ const V2StrategyMetricsChart: React.FC<{

const filteredDaily =
dailyMetrics
?.filter((m) => chartStrategyNames.includes(m.strategy.name))
?.filter((m) =>
chartStrategyNames.includes(m.strategy.name as StrategyKey)
)
.sort((a, b) => parseInt(a.timestamp) - parseInt(b.timestamp)) ?? [];

const filteredHourly =
hourlyMetrics
?.filter((m) => chartStrategyNames.includes(m.strategy.name))
?.filter((m) =>
chartStrategyNames.includes(m.strategy.name as StrategyKey)
)
.sort((a, b) => parseInt(a.timestamp) - parseInt(b.timestamp)) ?? [];

// if we are rendering chart for only one strategy we can use data as is
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@ export const Dashboards: DashboardData[] = [
title: 'TRV',
path: 'treasuryreservesvault',
description:
'Treasury Reserves Vault (TRV) coordinates and manages the flow of capital for current Treasury allocations. When funding and management parameters are approved for a Strategy, the TRV will transfer funds e.g. DAI and issue corresponding debt to the Strategy borrower. The current equity of the Strategy is discounted by the loan principal and accrued interest benchmarked to the prevailing rate of the current Base Strategy for the borrowed token.',
'Treasury Reserves Vault (TRV) is the central clearinghouse for the Temple Treasury and critical coordinator for current Treasury Strategy allocations. When funding and management parameters are approved for a Strategy, the TRV will transfer funds and issue corresponding debt to the Strategy borrower. The current equity of the Strategy is discounted by the loan principal and accrued interest benchmarked to the prevailing rate of the current Base Strategy for the borrowed token.',
contractLink: `${env.etherscan}/address/${env.contracts.treasuryReservesVault}`,
},
{
Expand All @@ -47,7 +47,7 @@ export const Dashboards: DashboardData[] = [
title: 'RAMOS',
path: 'ramos',
description:
'Ramos is the automated market operations (AMO) manager that supplies liquidity to the TEMPLE/DAI pool on the Balancer Exchange platform. A bot manages the contract to support TEMPLE trading, reduce price volatility, and earn farming rewards.',
'Ramos is the automated market operations (AMO) manager that supplies liquidity to the TEMPLE/DAI pool on the Balancer DEX. A bot manages the contract to support TEMPLE trading, reduce price volatility, and earn farming rewards.',
contractLink: `${env.etherscan}/address/${env.contracts.strategies.ramosStrategy}`,
},
{
Expand All @@ -56,7 +56,7 @@ export const Dashboards: DashboardData[] = [
title: 'TLC',
path: 'tlc',
description:
'Temple Loving Care (also known as Temple Line of Credit) offers DAI lending for users who supply TEMPLE token as collateral. The value of the collateral is not determined by the current $TEMPLE spot price on the Balancer DEX but by the current Treasury Price Index (TPI). Users may borrow up to 75% loan-to-value (LTV) with the liquidation LTV set to 80%. There are no origination fees and users can withdraw their collateral at any time by repaying the DAI loan. TLC interest rate is a variable APR that is dependent on Debt Ceiling Utilisation. Any accrued interest will increase LTV over time. Borrowers can expect the APR to be set no lower than the prevailing APR for the Treasury DAI Base Strategy. <a target="_blank" href="https://templedao.medium.com/he-who-controls-the-spice-controls-the-universe-bae5fb92bd43">Click here</a> to learn more about Temple Loving Care.',
'Temple Loving Care (also known as Temple Line of Credit) offers DAI lending for users who supply TEMPLE as collateral. The TLC will use the current Treasury Price Index (TPI) Oracle to determine the collateral value of TEMPLE. Users may borrow up to 85% loan-to-value (LTV) with the TEMPLE liquidation LTV set to 90%. There are no origination fees and users can withdraw their TEMPLE at any time by repaying the DAI loan. The TLC interest rate is set to a fixed rate that will be periodically updated to 2X the yield from the current Treasury Base Strategy e.g. sDAI. <a target="_blank" href="https://templedao.medium.com/he-who-controls-the-spice-controls-the-universe-bae5fb92bd43">Click here</a> to learn more about Temple Loving Care.',
contractLink: `${env.etherscan}/address/${env.contracts.strategies.tlcStrategy}`,
},
{
Expand All @@ -65,7 +65,7 @@ export const Dashboards: DashboardData[] = [
title: 'TEMPLE BASE',
path: 'templebase',
description:
'Temple Base strategy is the source of automated market operations (AMO) TEMPLE tokens in the Treasury framework. The TRV facilitates the withdrawal of newly minted TEMPLE tokens from and the issuance of TEMPLE debt to the Temple Base strategy. These TEMPLE tokens will be borrowed by a Treasury Strategy such as Ramos to generate returns. Once these tokens are repaid to the TRV, they will be deposited to the Temple Base strategy to be burned. From the perspective of the TRV, positive returns will be realized when TEMPLE flows to the Temple Base strategy is net positive.',
'TEMPLE Base strategy is the funding source for TEMPLE tokens for automated market operations (AMO) in the Treasury framework. The TRV facilitates the withdrawal of newly minted TEMPLE tokens from and the issuance of TEMPLE debt to the TEMPLE Base strategy. These TEMPLE tokens will be borrowed by a Treasury Strategy such as Ramos to generate returns. Once these tokens are repaid to the TRV, they will be deposited to the TEMPLE Base strategy to be burned. Positive returns will be realized when TEMPLE flows to the TEMPLE Base strategy is net positive.',
contractLink: `${env.etherscan}/address/${env.contracts.strategies.templeStrategy}`,
},
{
Expand All @@ -74,7 +74,7 @@ export const Dashboards: DashboardData[] = [
title: 'DSR BASE',
path: 'dsrbase',
description:
'Idle capital in the Treasury Reserves Vault (TRV) that is not currently deployed to a Strategy borrower will be automatically directed to a Base Strategy to earn yield. Currently, the Base Strategy is set to the Dai Savings Rate (DSR) which makes DAI the base currency of the TRV. The current rate of return for DSR Base also serves as the benchmark interest rate for the Treasury Strategy that borrows DAI from the TRV.',
'Idle reserve capital in the TRV that is not currently borrowed by a Strategy Borrower will be automatically directed to a Base Strategy to earn yield. The TRV Base Strategy is currently set to the Dai Savings Rate (DSR) or sDAI. The current rate of return for the Base Strategy also serves as the performance benchmark or "risk-free" interest rate for Treasury Strategies.',
contractLink: `${env.etherscan}/address/${env.contracts.strategies.dsrBaseStrategy}`,
},
{
Expand All @@ -83,7 +83,7 @@ export const Dashboards: DashboardData[] = [
title: 'TEMPLO MAYOR',
path: 'templomayor',
description:
'Templo Mayor is an Gnosis Safe Omnibus strategy. An Omnibus Strategy utilises the same bookkeeping structure and approval process, but may entail several related holdings or sub-positions that are managed as a whole. For instance, deposits into different but similar or co-dependent vaults on the same platform or different platforms may be consolidated into one Omnibus Gnosis Safe. Seed allocations of a target risk profile may also be consolidated into an Omnibus Strategy to reduce the noise. Therefore an Omnibus Strategy may provide additional operational efficiency and allow Stakeholders to evaluate a series of related deployments as one composite position rather than as singletons.',
'Templo Mayor is an Gnosis Safe Omnibus Strategy that is particularly useful when full automation is not feasible. An Omnibus Strategy utilises the same bookkeeping structure and approval process as the automated Temple v2 Strategies, but may entail several related holdings or sub-positions that are managed holistically. For instance, deposits into different but similar or co-dependent vaults on the same platform or different platforms may be consolidated into one Omnibus Gnosis Safe. Partner seed allocations of a target risk profile may also be consolidated into an Omnibus Strategy to derisk any particular project. An Omnibus Strategy may provide additional operational efficiency and allow Stakeholders to evaluate a series of related deployments as one composite position rather than as singletons.',
contractLink: `${env.etherscan}/address/${env.contracts.strategies.temploMayorGnosisStrategy}`,
},
{
Expand All @@ -92,7 +92,7 @@ export const Dashboards: DashboardData[] = [
title: 'FOHMO',
path: 'fohmo',
description:
'FOHMO is a strategy that aims to maintain a maxed looped position in OHM',
'FOHMO is a strategy that aims to maintain a maximally looped position in OHM',
contractLink: `${env.etherscan}/address/${env.contracts.strategies.fohmoGnosisStrategy}`,
},
];
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,12 @@ import { queryMinTablet } from 'styles/breakpoints';
import env from 'constants/env';
import linkSvg from 'assets/icons/link.svg?react';
import { formatNumberWithCommas } from 'utils/formatter';
import { DashboardData, Dashboards, isTRVDashboard } from '../DashboardConfig';
import {
DashboardData,
Dashboards,
isTRVDashboard,
StrategyKey,
} from '../DashboardConfig';

type Props = {
dashboardData: DashboardData;
Expand Down Expand Up @@ -271,8 +276,8 @@ const TxnHistoryTable = (props: Props) => {
const timeOnly = format(new Date(Number(tx.timestamp) * 1000), 'H:mm:ss');
return {
date: isBiggerThanTablet ? datetime : dateOnly,
type: tx.name,
strategy: tx.strategy.name,
type: tx.name as TxType,
strategy: tx.strategy.name as StrategyKey,
token: tx.token.symbol,
amount: amountFmt,
txHash: tx.hash,
Expand Down
Loading
Loading