@defisaver/positions-sdk 2.1.11 → 2.1.13
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.mocharc.json +4 -4
- package/.nvmrc +1 -1
- package/README.md +64 -64
- package/cjs/aaveV3/index.js +1 -1
- package/cjs/contracts.d.ts +100947 -134653
- package/cjs/helpers/morphoBlueHelpers/index.js +66 -66
- package/cjs/markets/aave/marketAssets.js +1 -1
- package/cjs/services/viem.d.ts +2 -3254
- package/cjs/staking/staking.d.ts +1 -1
- package/cjs/staking/staking.js +17 -14
- package/esm/aaveV3/index.js +1 -1
- package/esm/contracts.d.ts +100947 -134653
- package/esm/helpers/morphoBlueHelpers/index.js +66 -66
- package/esm/markets/aave/marketAssets.js +1 -1
- package/esm/services/viem.d.ts +2 -3254
- package/esm/staking/staking.d.ts +1 -1
- package/esm/staking/staking.js +17 -14
- package/package.json +47 -47
- package/src/aaveV2/index.ts +240 -240
- package/src/aaveV3/index.ts +614 -614
- package/src/aaveV3/merit.ts +97 -97
- package/src/aaveV3/merkl.ts +74 -74
- package/src/claiming/aaveV3.ts +154 -154
- package/src/claiming/compV3.ts +22 -22
- package/src/claiming/index.ts +12 -12
- package/src/claiming/king.ts +66 -66
- package/src/claiming/morphoBlue.ts +118 -118
- package/src/claiming/spark.ts +225 -225
- package/src/compoundV2/index.ts +244 -244
- package/src/compoundV3/index.ts +274 -274
- package/src/config/contracts.ts +1251 -1251
- package/src/constants/index.ts +10 -10
- package/src/contracts.ts +120 -120
- package/src/curveUsd/index.ts +254 -254
- package/src/eulerV2/index.ts +324 -324
- package/src/exchange/index.ts +25 -25
- package/src/fluid/index.ts +1668 -1668
- package/src/helpers/aaveHelpers/index.ts +187 -187
- package/src/helpers/compoundHelpers/index.ts +283 -283
- package/src/helpers/curveUsdHelpers/index.ts +40 -40
- package/src/helpers/eulerHelpers/index.ts +222 -222
- package/src/helpers/fluidHelpers/index.ts +326 -326
- package/src/helpers/index.ts +10 -10
- package/src/helpers/liquityV2Helpers/index.ts +82 -82
- package/src/helpers/llamaLendHelpers/index.ts +53 -53
- package/src/helpers/makerHelpers/index.ts +52 -52
- package/src/helpers/morphoBlueHelpers/index.ts +396 -396
- package/src/helpers/sparkHelpers/index.ts +158 -158
- package/src/index.ts +47 -47
- package/src/liquity/index.ts +159 -159
- package/src/liquityV2/index.ts +657 -657
- package/src/llamaLend/index.ts +305 -305
- package/src/maker/index.ts +223 -223
- package/src/markets/aave/index.ts +116 -116
- package/src/markets/aave/marketAssets.ts +54 -54
- package/src/markets/compound/index.ts +238 -238
- package/src/markets/compound/marketsAssets.ts +97 -97
- package/src/markets/curveUsd/index.ts +69 -69
- package/src/markets/euler/index.ts +26 -26
- package/src/markets/fluid/index.ts +2460 -2460
- package/src/markets/index.ts +25 -25
- package/src/markets/liquityV2/index.ts +102 -102
- package/src/markets/llamaLend/contractAddresses.ts +141 -141
- package/src/markets/llamaLend/index.ts +235 -235
- package/src/markets/morphoBlue/index.ts +895 -895
- package/src/markets/spark/index.ts +29 -29
- package/src/markets/spark/marketAssets.ts +12 -12
- package/src/moneymarket/moneymarketCommonService.ts +80 -80
- package/src/morphoBlue/index.ts +274 -274
- package/src/portfolio/index.ts +570 -570
- package/src/services/priceService.ts +159 -159
- package/src/services/utils.ts +115 -115
- package/src/services/viem.ts +34 -34
- package/src/setup.ts +8 -8
- package/src/spark/index.ts +456 -456
- package/src/staking/eligibility.ts +53 -53
- package/src/staking/index.ts +1 -1
- package/src/staking/staking.ts +177 -172
- package/src/types/aave.ts +189 -189
- package/src/types/claiming.ts +109 -109
- package/src/types/common.ts +107 -107
- package/src/types/compound.ts +136 -136
- package/src/types/curveUsd.ts +123 -123
- package/src/types/euler.ts +175 -175
- package/src/types/fluid.ts +452 -452
- package/src/types/index.ts +13 -13
- package/src/types/liquity.ts +30 -30
- package/src/types/liquityV2.ts +126 -126
- package/src/types/llamaLend.ts +159 -159
- package/src/types/maker.ts +63 -63
- package/src/types/merit.ts +1 -1
- package/src/types/merkl.ts +70 -70
- package/src/types/morphoBlue.ts +194 -194
- package/src/types/portfolio.ts +60 -60
- package/src/types/spark.ts +133 -133
- package/src/umbrella/index.ts +69 -69
- package/src/umbrella/umbrellaUtils.ts +29 -29
package/src/compoundV2/index.ts
CHANGED
|
@@ -1,244 +1,244 @@
|
|
|
1
|
-
import { assetAmountInEth, getAssetInfo, getAssetInfoByAddress } from '@defisaver/tokens';
|
|
2
|
-
import Dec from 'decimal.js';
|
|
3
|
-
import { Client } from 'viem';
|
|
4
|
-
import { BLOCKS_IN_A_YEAR } from '../constants';
|
|
5
|
-
import { aprToApy } from '../moneymarket';
|
|
6
|
-
import { compareAddresses, handleWbtcLegacy, wethToEth } from '../services/utils';
|
|
7
|
-
import {
|
|
8
|
-
Blockish, EthAddress, EthereumProvider, IncentiveKind, NetworkNumber, PositionBalances,
|
|
9
|
-
} from '../types/common';
|
|
10
|
-
import { CompoundLoanInfoContractViem, ComptrollerContractViem } from '../contracts';
|
|
11
|
-
import { compoundV2CollateralAssets } from '../markets';
|
|
12
|
-
import {
|
|
13
|
-
CompoundV2AssetsData, CompoundV2MarketsData, CompoundV2PositionData, CompoundV2UsedAsset, CompoundV2UsedAssets,
|
|
14
|
-
} from '../types';
|
|
15
|
-
import { getCompoundV2AggregatedData } from '../helpers/compoundHelpers';
|
|
16
|
-
import { getViemProvider, setViemBlockNumber } from '../services/viem';
|
|
17
|
-
|
|
18
|
-
const compAddress = '0xc00e94cb662c3520282e6f5717214004a7f26888';
|
|
19
|
-
|
|
20
|
-
export const _getCompoundV2MarketsData = async (provider: Client, network: NetworkNumber): Promise<CompoundV2MarketsData> => {
|
|
21
|
-
const cAddresses = compoundV2CollateralAssets.map(a => a.address);
|
|
22
|
-
|
|
23
|
-
const loanInfoContract = CompoundLoanInfoContractViem(provider, network);
|
|
24
|
-
const loanInfo = await loanInfoContract.read.getFullTokensInfo([cAddresses as EthAddress[]]);
|
|
25
|
-
|
|
26
|
-
const compPrice = loanInfo.find(m => compareAddresses(m.underlyingTokenAddress, compAddress))!.price.toString();
|
|
27
|
-
|
|
28
|
-
const assetsData = loanInfo
|
|
29
|
-
.map((market, i) => {
|
|
30
|
-
let symbol = getAssetInfoByAddress(cAddresses[i]).underlyingAsset;
|
|
31
|
-
let isWbtcLegacy = false;
|
|
32
|
-
const totalSupply = new Dec(market.totalSupply.toString()).div(1e18).times(market.exchangeRate.toString());
|
|
33
|
-
const totalBorrow = market.totalBorrow.toString();
|
|
34
|
-
const borrowCap = market.borrowCap.toString();
|
|
35
|
-
const compSupplySpeeds = market.compSupplySpeeds.toString();
|
|
36
|
-
const compBorrowSpeeds = market.compBorrowSpeeds.toString();
|
|
37
|
-
const assetPrice = market.price.toString();
|
|
38
|
-
|
|
39
|
-
const pricePrecisionDiff = 18 - getAssetInfo(getAssetInfoByAddress(cAddresses[i]).underlyingAsset).decimals;
|
|
40
|
-
|
|
41
|
-
if (cAddresses[i].toLowerCase() === '0xc11b1268c1a384e55c48c2391d8d480264a3a7f4'.toLowerCase()) {
|
|
42
|
-
symbol = 'WBTC Legacy';
|
|
43
|
-
isWbtcLegacy = true;
|
|
44
|
-
}
|
|
45
|
-
return {
|
|
46
|
-
symbol,
|
|
47
|
-
underlyingTokenAddress: market.underlyingTokenAddress,
|
|
48
|
-
supplyRate: aprToApy(new Dec(BLOCKS_IN_A_YEAR).times(market.supplyRate.toString()).div(1e16).toString()).toString(),
|
|
49
|
-
borrowRate: aprToApy(new Dec(BLOCKS_IN_A_YEAR).times(market.borrowRate.toString()).div(1e16).toString()).toString(),
|
|
50
|
-
supplyIncentives: [{
|
|
51
|
-
token: 'COMP',
|
|
52
|
-
apy: aprToApy((100 * BLOCKS_IN_A_YEAR * +compSupplySpeeds * +compPrice) / +assetPrice / +totalSupply).toString(),
|
|
53
|
-
incentiveKind: IncentiveKind.Reward,
|
|
54
|
-
description: 'Eligible for protocol-level COMP incentives.',
|
|
55
|
-
}],
|
|
56
|
-
borrowIncentives: [{
|
|
57
|
-
token: 'COMP',
|
|
58
|
-
apy: aprToApy((100 * BLOCKS_IN_A_YEAR * +compBorrowSpeeds * +compPrice) / +assetPrice / +totalBorrow).toString(),
|
|
59
|
-
incentiveKind: IncentiveKind.Reward,
|
|
60
|
-
description: 'Eligible for protocol-level COMP incentives.',
|
|
61
|
-
}],
|
|
62
|
-
collateralFactor: new Dec(market.collateralFactor.toString()).div(1e18).toString(),
|
|
63
|
-
marketLiquidity: assetAmountInEth(market.marketLiquidity.toString(), handleWbtcLegacy(symbol)),
|
|
64
|
-
utilization: new Dec(market.totalBorrow.toString()).div(totalSupply).times(100).toString(),
|
|
65
|
-
totalSupply: assetAmountInEth(totalSupply, handleWbtcLegacy(symbol)),
|
|
66
|
-
totalBorrow: assetAmountInEth(totalBorrow, handleWbtcLegacy(symbol)),
|
|
67
|
-
exchangeRate: new Dec(market.exchangeRate.toString()).div(1e28).toString(),
|
|
68
|
-
borrowCap: assetAmountInEth(borrowCap, handleWbtcLegacy(symbol)),
|
|
69
|
-
canBeBorrowed: market.canBorrow,
|
|
70
|
-
canBeSupplied: market.canMint,
|
|
71
|
-
price: new Dec(market.price.toString()).div(10 ** (18 + pricePrecisionDiff)).toString(),
|
|
72
|
-
};
|
|
73
|
-
});
|
|
74
|
-
|
|
75
|
-
const payload = {} as CompoundV2AssetsData;
|
|
76
|
-
// Sort by market size
|
|
77
|
-
assetsData
|
|
78
|
-
.sort((a, b) => {
|
|
79
|
-
const aMarket = new Dec(a.price).times(a.totalSupply).toString();
|
|
80
|
-
const bMarket = new Dec(b.price).times(b.totalSupply).toString();
|
|
81
|
-
|
|
82
|
-
return new Dec(bMarket).minus(aMarket).toNumber();
|
|
83
|
-
})
|
|
84
|
-
.forEach((market, i) => { payload[market.symbol] = { ...market, sortIndex: i }; });
|
|
85
|
-
|
|
86
|
-
return { assetsData: payload };
|
|
87
|
-
};
|
|
88
|
-
|
|
89
|
-
export const getCompoundV2MarketsData = async (provider: EthereumProvider, network: NetworkNumber): Promise<CompoundV2MarketsData> => _getCompoundV2MarketsData(getViemProvider(provider, network), network);
|
|
90
|
-
|
|
91
|
-
export const EMPTY_COMPOUND_DATA = {
|
|
92
|
-
usedAssets: {},
|
|
93
|
-
suppliedUsd: '0',
|
|
94
|
-
borrowedUsd: '0',
|
|
95
|
-
borrowLimitUsd: '0',
|
|
96
|
-
leftToBorrowUsd: '0',
|
|
97
|
-
ratio: '0',
|
|
98
|
-
minRatio: '0',
|
|
99
|
-
netApy: '0',
|
|
100
|
-
incentiveUsd: '0',
|
|
101
|
-
totalInterestUsd: '0',
|
|
102
|
-
borrowStableSupplyUnstable: false,
|
|
103
|
-
};
|
|
104
|
-
|
|
105
|
-
const getCollateralAssetsAddresses = async (provider: Client, network: NetworkNumber, account: EthAddress) => {
|
|
106
|
-
const contract = ComptrollerContractViem(provider, network);
|
|
107
|
-
|
|
108
|
-
return contract.read.getAssetsIn([account]);
|
|
109
|
-
};
|
|
110
|
-
|
|
111
|
-
const getAllMarketAddresses = async (provider: Client, network: NetworkNumber, block: Blockish) => {
|
|
112
|
-
const contract = ComptrollerContractViem(provider, network);
|
|
113
|
-
|
|
114
|
-
return contract.read.getAllMarkets(setViemBlockNumber(block));
|
|
115
|
-
};
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
export const _getCompoundV2AccountBalances = async (provider: Client, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress): Promise<PositionBalances> => {
|
|
119
|
-
let balances: PositionBalances = {
|
|
120
|
-
collateral: {},
|
|
121
|
-
debt: {},
|
|
122
|
-
};
|
|
123
|
-
|
|
124
|
-
if (!address) {
|
|
125
|
-
return balances;
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
const assets = await getAllMarketAddresses(provider, network, block);
|
|
129
|
-
const assetInfo = assets.map(a => getAssetInfoByAddress(a, network));
|
|
130
|
-
const loanInfoContract = CompoundLoanInfoContractViem(provider, network, block);
|
|
131
|
-
const loanInfo = await loanInfoContract.read.getTokenBalances([address, assets], setViemBlockNumber(block));
|
|
132
|
-
|
|
133
|
-
loanInfo[0].forEach((weiAmount: any, i: number) => {
|
|
134
|
-
const asset = wethToEth(
|
|
135
|
-
assetInfo[i].symbol === 'cWBTC Legacy'
|
|
136
|
-
? `${assetInfo[i].underlyingAsset} Legacy`
|
|
137
|
-
: assetInfo[i].underlyingAsset,
|
|
138
|
-
);
|
|
139
|
-
|
|
140
|
-
balances = {
|
|
141
|
-
collateral: {
|
|
142
|
-
...balances.collateral,
|
|
143
|
-
[addressMapping ? getAssetInfo(asset, network).address.toLowerCase() : asset]: weiAmount.toString(),
|
|
144
|
-
},
|
|
145
|
-
};
|
|
146
|
-
});
|
|
147
|
-
loanInfo[1].forEach((weiAmount: any, i: number) => {
|
|
148
|
-
const asset = wethToEth(
|
|
149
|
-
assetInfo[i].symbol === 'cWBTC Legacy'
|
|
150
|
-
? `${assetInfo[i].underlyingAsset} Legacy`
|
|
151
|
-
: assetInfo[i].underlyingAsset,
|
|
152
|
-
);
|
|
153
|
-
|
|
154
|
-
balances = {
|
|
155
|
-
...balances,
|
|
156
|
-
debt: {
|
|
157
|
-
...balances.debt,
|
|
158
|
-
[addressMapping ? getAssetInfo(asset, network).address.toLowerCase() : asset]: weiAmount.toString(),
|
|
159
|
-
},
|
|
160
|
-
};
|
|
161
|
-
});
|
|
162
|
-
|
|
163
|
-
return balances;
|
|
164
|
-
};
|
|
165
|
-
|
|
166
|
-
export const getCompoundV2AccountBalances = async (
|
|
167
|
-
provider: EthereumProvider,
|
|
168
|
-
network: NetworkNumber,
|
|
169
|
-
block: Blockish,
|
|
170
|
-
addressMapping: boolean,
|
|
171
|
-
address: EthAddress,
|
|
172
|
-
): Promise<PositionBalances> => _getCompoundV2AccountBalances(getViemProvider(provider, network), network, block, addressMapping, address);
|
|
173
|
-
|
|
174
|
-
export const _getCompoundV2AccountData = async (provider: Client, network: NetworkNumber, address: EthAddress, assetsData: CompoundV2AssetsData): Promise<CompoundV2PositionData> => {
|
|
175
|
-
if (!address) throw new Error('No address provided');
|
|
176
|
-
|
|
177
|
-
let payload = { ...EMPTY_COMPOUND_DATA, lastUpdated: Date.now() };
|
|
178
|
-
|
|
179
|
-
const loanInfoContract = CompoundLoanInfoContractViem(provider, network);
|
|
180
|
-
const [loanInfo, collateralAssetsAddresses] = await Promise.all([
|
|
181
|
-
loanInfoContract.read.getTokenBalances([address, compoundV2CollateralAssets.map(a => a.address) as EthAddress[]]),
|
|
182
|
-
getCollateralAssetsAddresses(provider, network, address),
|
|
183
|
-
]);
|
|
184
|
-
|
|
185
|
-
const usedAssets = {} as CompoundV2UsedAssets;
|
|
186
|
-
|
|
187
|
-
loanInfo[0].forEach((weiAmount: BigInt, i: number) => {
|
|
188
|
-
const asset = compoundV2CollateralAssets[i].symbol === 'cWBTC Legacy'
|
|
189
|
-
? `${compoundV2CollateralAssets[i].underlyingAsset} Legacy`
|
|
190
|
-
: compoundV2CollateralAssets[i].underlyingAsset;
|
|
191
|
-
const amount = assetAmountInEth(weiAmount.toString(), asset);
|
|
192
|
-
const collateral = !!collateralAssetsAddresses.find(a => compareAddresses(a, compoundV2CollateralAssets[i].address));
|
|
193
|
-
if (weiAmount.toString() === '0' && !collateral) return;
|
|
194
|
-
if (!usedAssets[asset]) usedAssets[asset] = {} as CompoundV2UsedAsset;
|
|
195
|
-
usedAssets[asset] = {
|
|
196
|
-
...usedAssets[asset],
|
|
197
|
-
symbol: asset,
|
|
198
|
-
supplied: amount,
|
|
199
|
-
suppliedUsd: new Dec(amount).mul(assetsData[handleWbtcLegacy(asset)].price).toString(),
|
|
200
|
-
isSupplied: +amount > 0,
|
|
201
|
-
collateral,
|
|
202
|
-
};
|
|
203
|
-
});
|
|
204
|
-
|
|
205
|
-
loanInfo[1].forEach((weiAmount, i) => {
|
|
206
|
-
if (weiAmount.toString() === '0') return;
|
|
207
|
-
const asset = compoundV2CollateralAssets[i].symbol === 'cWBTC Legacy'
|
|
208
|
-
? `${compoundV2CollateralAssets[i].underlyingAsset} Legacy`
|
|
209
|
-
: compoundV2CollateralAssets[i].underlyingAsset;
|
|
210
|
-
const amount = assetAmountInEth(weiAmount.toString(), asset);
|
|
211
|
-
if (!usedAssets[asset]) usedAssets[asset] = {} as CompoundV2UsedAsset;
|
|
212
|
-
usedAssets[asset] = {
|
|
213
|
-
...usedAssets[asset],
|
|
214
|
-
symbol: asset,
|
|
215
|
-
borrowed: amount,
|
|
216
|
-
borrowedUsd: new Dec(amount).mul(assetsData[handleWbtcLegacy(asset)].price).toString(),
|
|
217
|
-
isBorrowed: true,
|
|
218
|
-
collateral: !!usedAssets[asset].collateral,
|
|
219
|
-
};
|
|
220
|
-
});
|
|
221
|
-
|
|
222
|
-
payload = {
|
|
223
|
-
...payload,
|
|
224
|
-
usedAssets,
|
|
225
|
-
...getCompoundV2AggregatedData({
|
|
226
|
-
usedAssets, assetsData,
|
|
227
|
-
}),
|
|
228
|
-
};
|
|
229
|
-
|
|
230
|
-
return payload;
|
|
231
|
-
};
|
|
232
|
-
|
|
233
|
-
export const getCompoundV2AccountData = async (
|
|
234
|
-
provider: EthereumProvider,
|
|
235
|
-
network: NetworkNumber,
|
|
236
|
-
address: EthAddress,
|
|
237
|
-
assetsData: CompoundV2AssetsData,
|
|
238
|
-
): Promise<CompoundV2PositionData> => _getCompoundV2AccountData(getViemProvider(provider, network), network, address, assetsData);
|
|
239
|
-
|
|
240
|
-
export const getCompoundV2FullPositionData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress): Promise<CompoundV2PositionData> => {
|
|
241
|
-
const marketData = await getCompoundV2MarketsData(provider, network);
|
|
242
|
-
const positionData = await getCompoundV2AccountData(provider, network, address, marketData.assetsData);
|
|
243
|
-
return positionData;
|
|
244
|
-
};
|
|
1
|
+
import { assetAmountInEth, getAssetInfo, getAssetInfoByAddress } from '@defisaver/tokens';
|
|
2
|
+
import Dec from 'decimal.js';
|
|
3
|
+
import { Client } from 'viem';
|
|
4
|
+
import { BLOCKS_IN_A_YEAR } from '../constants';
|
|
5
|
+
import { aprToApy } from '../moneymarket';
|
|
6
|
+
import { compareAddresses, handleWbtcLegacy, wethToEth } from '../services/utils';
|
|
7
|
+
import {
|
|
8
|
+
Blockish, EthAddress, EthereumProvider, IncentiveKind, NetworkNumber, PositionBalances,
|
|
9
|
+
} from '../types/common';
|
|
10
|
+
import { CompoundLoanInfoContractViem, ComptrollerContractViem } from '../contracts';
|
|
11
|
+
import { compoundV2CollateralAssets } from '../markets';
|
|
12
|
+
import {
|
|
13
|
+
CompoundV2AssetsData, CompoundV2MarketsData, CompoundV2PositionData, CompoundV2UsedAsset, CompoundV2UsedAssets,
|
|
14
|
+
} from '../types';
|
|
15
|
+
import { getCompoundV2AggregatedData } from '../helpers/compoundHelpers';
|
|
16
|
+
import { getViemProvider, setViemBlockNumber } from '../services/viem';
|
|
17
|
+
|
|
18
|
+
const compAddress = '0xc00e94cb662c3520282e6f5717214004a7f26888';
|
|
19
|
+
|
|
20
|
+
export const _getCompoundV2MarketsData = async (provider: Client, network: NetworkNumber): Promise<CompoundV2MarketsData> => {
|
|
21
|
+
const cAddresses = compoundV2CollateralAssets.map(a => a.address);
|
|
22
|
+
|
|
23
|
+
const loanInfoContract = CompoundLoanInfoContractViem(provider, network);
|
|
24
|
+
const loanInfo = await loanInfoContract.read.getFullTokensInfo([cAddresses as EthAddress[]]);
|
|
25
|
+
|
|
26
|
+
const compPrice = loanInfo.find(m => compareAddresses(m.underlyingTokenAddress, compAddress))!.price.toString();
|
|
27
|
+
|
|
28
|
+
const assetsData = loanInfo
|
|
29
|
+
.map((market, i) => {
|
|
30
|
+
let symbol = getAssetInfoByAddress(cAddresses[i]).underlyingAsset;
|
|
31
|
+
let isWbtcLegacy = false;
|
|
32
|
+
const totalSupply = new Dec(market.totalSupply.toString()).div(1e18).times(market.exchangeRate.toString());
|
|
33
|
+
const totalBorrow = market.totalBorrow.toString();
|
|
34
|
+
const borrowCap = market.borrowCap.toString();
|
|
35
|
+
const compSupplySpeeds = market.compSupplySpeeds.toString();
|
|
36
|
+
const compBorrowSpeeds = market.compBorrowSpeeds.toString();
|
|
37
|
+
const assetPrice = market.price.toString();
|
|
38
|
+
|
|
39
|
+
const pricePrecisionDiff = 18 - getAssetInfo(getAssetInfoByAddress(cAddresses[i]).underlyingAsset).decimals;
|
|
40
|
+
|
|
41
|
+
if (cAddresses[i].toLowerCase() === '0xc11b1268c1a384e55c48c2391d8d480264a3a7f4'.toLowerCase()) {
|
|
42
|
+
symbol = 'WBTC Legacy';
|
|
43
|
+
isWbtcLegacy = true;
|
|
44
|
+
}
|
|
45
|
+
return {
|
|
46
|
+
symbol,
|
|
47
|
+
underlyingTokenAddress: market.underlyingTokenAddress,
|
|
48
|
+
supplyRate: aprToApy(new Dec(BLOCKS_IN_A_YEAR).times(market.supplyRate.toString()).div(1e16).toString()).toString(),
|
|
49
|
+
borrowRate: aprToApy(new Dec(BLOCKS_IN_A_YEAR).times(market.borrowRate.toString()).div(1e16).toString()).toString(),
|
|
50
|
+
supplyIncentives: [{
|
|
51
|
+
token: 'COMP',
|
|
52
|
+
apy: aprToApy((100 * BLOCKS_IN_A_YEAR * +compSupplySpeeds * +compPrice) / +assetPrice / +totalSupply).toString(),
|
|
53
|
+
incentiveKind: IncentiveKind.Reward,
|
|
54
|
+
description: 'Eligible for protocol-level COMP incentives.',
|
|
55
|
+
}],
|
|
56
|
+
borrowIncentives: [{
|
|
57
|
+
token: 'COMP',
|
|
58
|
+
apy: aprToApy((100 * BLOCKS_IN_A_YEAR * +compBorrowSpeeds * +compPrice) / +assetPrice / +totalBorrow).toString(),
|
|
59
|
+
incentiveKind: IncentiveKind.Reward,
|
|
60
|
+
description: 'Eligible for protocol-level COMP incentives.',
|
|
61
|
+
}],
|
|
62
|
+
collateralFactor: new Dec(market.collateralFactor.toString()).div(1e18).toString(),
|
|
63
|
+
marketLiquidity: assetAmountInEth(market.marketLiquidity.toString(), handleWbtcLegacy(symbol)),
|
|
64
|
+
utilization: new Dec(market.totalBorrow.toString()).div(totalSupply).times(100).toString(),
|
|
65
|
+
totalSupply: assetAmountInEth(totalSupply, handleWbtcLegacy(symbol)),
|
|
66
|
+
totalBorrow: assetAmountInEth(totalBorrow, handleWbtcLegacy(symbol)),
|
|
67
|
+
exchangeRate: new Dec(market.exchangeRate.toString()).div(1e28).toString(),
|
|
68
|
+
borrowCap: assetAmountInEth(borrowCap, handleWbtcLegacy(symbol)),
|
|
69
|
+
canBeBorrowed: market.canBorrow,
|
|
70
|
+
canBeSupplied: market.canMint,
|
|
71
|
+
price: new Dec(market.price.toString()).div(10 ** (18 + pricePrecisionDiff)).toString(),
|
|
72
|
+
};
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
const payload = {} as CompoundV2AssetsData;
|
|
76
|
+
// Sort by market size
|
|
77
|
+
assetsData
|
|
78
|
+
.sort((a, b) => {
|
|
79
|
+
const aMarket = new Dec(a.price).times(a.totalSupply).toString();
|
|
80
|
+
const bMarket = new Dec(b.price).times(b.totalSupply).toString();
|
|
81
|
+
|
|
82
|
+
return new Dec(bMarket).minus(aMarket).toNumber();
|
|
83
|
+
})
|
|
84
|
+
.forEach((market, i) => { payload[market.symbol] = { ...market, sortIndex: i }; });
|
|
85
|
+
|
|
86
|
+
return { assetsData: payload };
|
|
87
|
+
};
|
|
88
|
+
|
|
89
|
+
export const getCompoundV2MarketsData = async (provider: EthereumProvider, network: NetworkNumber): Promise<CompoundV2MarketsData> => _getCompoundV2MarketsData(getViemProvider(provider, network), network);
|
|
90
|
+
|
|
91
|
+
export const EMPTY_COMPOUND_DATA = {
|
|
92
|
+
usedAssets: {},
|
|
93
|
+
suppliedUsd: '0',
|
|
94
|
+
borrowedUsd: '0',
|
|
95
|
+
borrowLimitUsd: '0',
|
|
96
|
+
leftToBorrowUsd: '0',
|
|
97
|
+
ratio: '0',
|
|
98
|
+
minRatio: '0',
|
|
99
|
+
netApy: '0',
|
|
100
|
+
incentiveUsd: '0',
|
|
101
|
+
totalInterestUsd: '0',
|
|
102
|
+
borrowStableSupplyUnstable: false,
|
|
103
|
+
};
|
|
104
|
+
|
|
105
|
+
const getCollateralAssetsAddresses = async (provider: Client, network: NetworkNumber, account: EthAddress) => {
|
|
106
|
+
const contract = ComptrollerContractViem(provider, network);
|
|
107
|
+
|
|
108
|
+
return contract.read.getAssetsIn([account]);
|
|
109
|
+
};
|
|
110
|
+
|
|
111
|
+
const getAllMarketAddresses = async (provider: Client, network: NetworkNumber, block: Blockish) => {
|
|
112
|
+
const contract = ComptrollerContractViem(provider, network);
|
|
113
|
+
|
|
114
|
+
return contract.read.getAllMarkets(setViemBlockNumber(block));
|
|
115
|
+
};
|
|
116
|
+
|
|
117
|
+
|
|
118
|
+
export const _getCompoundV2AccountBalances = async (provider: Client, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress): Promise<PositionBalances> => {
|
|
119
|
+
let balances: PositionBalances = {
|
|
120
|
+
collateral: {},
|
|
121
|
+
debt: {},
|
|
122
|
+
};
|
|
123
|
+
|
|
124
|
+
if (!address) {
|
|
125
|
+
return balances;
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
const assets = await getAllMarketAddresses(provider, network, block);
|
|
129
|
+
const assetInfo = assets.map(a => getAssetInfoByAddress(a, network));
|
|
130
|
+
const loanInfoContract = CompoundLoanInfoContractViem(provider, network, block);
|
|
131
|
+
const loanInfo = await loanInfoContract.read.getTokenBalances([address, assets], setViemBlockNumber(block));
|
|
132
|
+
|
|
133
|
+
loanInfo[0].forEach((weiAmount: any, i: number) => {
|
|
134
|
+
const asset = wethToEth(
|
|
135
|
+
assetInfo[i].symbol === 'cWBTC Legacy'
|
|
136
|
+
? `${assetInfo[i].underlyingAsset} Legacy`
|
|
137
|
+
: assetInfo[i].underlyingAsset,
|
|
138
|
+
);
|
|
139
|
+
|
|
140
|
+
balances = {
|
|
141
|
+
collateral: {
|
|
142
|
+
...balances.collateral,
|
|
143
|
+
[addressMapping ? getAssetInfo(asset, network).address.toLowerCase() : asset]: weiAmount.toString(),
|
|
144
|
+
},
|
|
145
|
+
};
|
|
146
|
+
});
|
|
147
|
+
loanInfo[1].forEach((weiAmount: any, i: number) => {
|
|
148
|
+
const asset = wethToEth(
|
|
149
|
+
assetInfo[i].symbol === 'cWBTC Legacy'
|
|
150
|
+
? `${assetInfo[i].underlyingAsset} Legacy`
|
|
151
|
+
: assetInfo[i].underlyingAsset,
|
|
152
|
+
);
|
|
153
|
+
|
|
154
|
+
balances = {
|
|
155
|
+
...balances,
|
|
156
|
+
debt: {
|
|
157
|
+
...balances.debt,
|
|
158
|
+
[addressMapping ? getAssetInfo(asset, network).address.toLowerCase() : asset]: weiAmount.toString(),
|
|
159
|
+
},
|
|
160
|
+
};
|
|
161
|
+
});
|
|
162
|
+
|
|
163
|
+
return balances;
|
|
164
|
+
};
|
|
165
|
+
|
|
166
|
+
export const getCompoundV2AccountBalances = async (
|
|
167
|
+
provider: EthereumProvider,
|
|
168
|
+
network: NetworkNumber,
|
|
169
|
+
block: Blockish,
|
|
170
|
+
addressMapping: boolean,
|
|
171
|
+
address: EthAddress,
|
|
172
|
+
): Promise<PositionBalances> => _getCompoundV2AccountBalances(getViemProvider(provider, network), network, block, addressMapping, address);
|
|
173
|
+
|
|
174
|
+
export const _getCompoundV2AccountData = async (provider: Client, network: NetworkNumber, address: EthAddress, assetsData: CompoundV2AssetsData): Promise<CompoundV2PositionData> => {
|
|
175
|
+
if (!address) throw new Error('No address provided');
|
|
176
|
+
|
|
177
|
+
let payload = { ...EMPTY_COMPOUND_DATA, lastUpdated: Date.now() };
|
|
178
|
+
|
|
179
|
+
const loanInfoContract = CompoundLoanInfoContractViem(provider, network);
|
|
180
|
+
const [loanInfo, collateralAssetsAddresses] = await Promise.all([
|
|
181
|
+
loanInfoContract.read.getTokenBalances([address, compoundV2CollateralAssets.map(a => a.address) as EthAddress[]]),
|
|
182
|
+
getCollateralAssetsAddresses(provider, network, address),
|
|
183
|
+
]);
|
|
184
|
+
|
|
185
|
+
const usedAssets = {} as CompoundV2UsedAssets;
|
|
186
|
+
|
|
187
|
+
loanInfo[0].forEach((weiAmount: BigInt, i: number) => {
|
|
188
|
+
const asset = compoundV2CollateralAssets[i].symbol === 'cWBTC Legacy'
|
|
189
|
+
? `${compoundV2CollateralAssets[i].underlyingAsset} Legacy`
|
|
190
|
+
: compoundV2CollateralAssets[i].underlyingAsset;
|
|
191
|
+
const amount = assetAmountInEth(weiAmount.toString(), asset);
|
|
192
|
+
const collateral = !!collateralAssetsAddresses.find(a => compareAddresses(a, compoundV2CollateralAssets[i].address));
|
|
193
|
+
if (weiAmount.toString() === '0' && !collateral) return;
|
|
194
|
+
if (!usedAssets[asset]) usedAssets[asset] = {} as CompoundV2UsedAsset;
|
|
195
|
+
usedAssets[asset] = {
|
|
196
|
+
...usedAssets[asset],
|
|
197
|
+
symbol: asset,
|
|
198
|
+
supplied: amount,
|
|
199
|
+
suppliedUsd: new Dec(amount).mul(assetsData[handleWbtcLegacy(asset)].price).toString(),
|
|
200
|
+
isSupplied: +amount > 0,
|
|
201
|
+
collateral,
|
|
202
|
+
};
|
|
203
|
+
});
|
|
204
|
+
|
|
205
|
+
loanInfo[1].forEach((weiAmount, i) => {
|
|
206
|
+
if (weiAmount.toString() === '0') return;
|
|
207
|
+
const asset = compoundV2CollateralAssets[i].symbol === 'cWBTC Legacy'
|
|
208
|
+
? `${compoundV2CollateralAssets[i].underlyingAsset} Legacy`
|
|
209
|
+
: compoundV2CollateralAssets[i].underlyingAsset;
|
|
210
|
+
const amount = assetAmountInEth(weiAmount.toString(), asset);
|
|
211
|
+
if (!usedAssets[asset]) usedAssets[asset] = {} as CompoundV2UsedAsset;
|
|
212
|
+
usedAssets[asset] = {
|
|
213
|
+
...usedAssets[asset],
|
|
214
|
+
symbol: asset,
|
|
215
|
+
borrowed: amount,
|
|
216
|
+
borrowedUsd: new Dec(amount).mul(assetsData[handleWbtcLegacy(asset)].price).toString(),
|
|
217
|
+
isBorrowed: true,
|
|
218
|
+
collateral: !!usedAssets[asset].collateral,
|
|
219
|
+
};
|
|
220
|
+
});
|
|
221
|
+
|
|
222
|
+
payload = {
|
|
223
|
+
...payload,
|
|
224
|
+
usedAssets,
|
|
225
|
+
...getCompoundV2AggregatedData({
|
|
226
|
+
usedAssets, assetsData,
|
|
227
|
+
}),
|
|
228
|
+
};
|
|
229
|
+
|
|
230
|
+
return payload;
|
|
231
|
+
};
|
|
232
|
+
|
|
233
|
+
export const getCompoundV2AccountData = async (
|
|
234
|
+
provider: EthereumProvider,
|
|
235
|
+
network: NetworkNumber,
|
|
236
|
+
address: EthAddress,
|
|
237
|
+
assetsData: CompoundV2AssetsData,
|
|
238
|
+
): Promise<CompoundV2PositionData> => _getCompoundV2AccountData(getViemProvider(provider, network), network, address, assetsData);
|
|
239
|
+
|
|
240
|
+
export const getCompoundV2FullPositionData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress): Promise<CompoundV2PositionData> => {
|
|
241
|
+
const marketData = await getCompoundV2MarketsData(provider, network);
|
|
242
|
+
const positionData = await getCompoundV2AccountData(provider, network, address, marketData.assetsData);
|
|
243
|
+
return positionData;
|
|
244
|
+
};
|