@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/compoundV3/index.ts
CHANGED
|
@@ -1,274 +1,274 @@
|
|
|
1
|
-
import Dec from 'decimal.js';
|
|
2
|
-
import {
|
|
3
|
-
assetAmountInEth, getAssetInfo, getAssetInfoByAddress,
|
|
4
|
-
} from '@defisaver/tokens';
|
|
5
|
-
import { Client } from 'viem';
|
|
6
|
-
import { CompV3ViewContractViem } from '../contracts';
|
|
7
|
-
import {
|
|
8
|
-
CompoundV3AssetData, CompoundMarketData, CompoundV3AssetsData, CompoundV3UsedAssets, CompoundV3MarketsData, CompoundV3PositionData,
|
|
9
|
-
} from '../types';
|
|
10
|
-
import {
|
|
11
|
-
Blockish, EthAddress, EthereumProvider, IncentiveKind, NetworkNumber, PositionBalances,
|
|
12
|
-
} from '../types/common';
|
|
13
|
-
import {
|
|
14
|
-
getStakingApy, STAKING_ASSETS,
|
|
15
|
-
} from '../staking';
|
|
16
|
-
import { ethToWeth, wethToEth } from '../services/utils';
|
|
17
|
-
import { ZERO_ADDRESS } from '../constants';
|
|
18
|
-
import { calculateBorrowingAssetLimit } from '../moneymarket';
|
|
19
|
-
import {
|
|
20
|
-
formatBaseData, formatMarketData, getCompoundV3AggregatedData, getIncentiveApys,
|
|
21
|
-
} from '../helpers/compoundHelpers';
|
|
22
|
-
import {
|
|
23
|
-
COMPOUND_V3_ETH, COMPOUND_V3_USDBC, COMPOUND_V3_USDC, COMPOUND_V3_USDCe, COMPOUND_V3_USDT,
|
|
24
|
-
} from '../markets/compound';
|
|
25
|
-
import {
|
|
26
|
-
getEthPrice, getCompPrice, getUSDCPrice, getWstETHPrice,
|
|
27
|
-
} from '../services/priceService';
|
|
28
|
-
import { getViemProvider, setViemBlockNumber } from '../services/viem';
|
|
29
|
-
|
|
30
|
-
const getSupportedAssetsAddressesForMarket = (selectedMarket: CompoundMarketData, network: NetworkNumber) => selectedMarket.collAssets.map(asset => getAssetInfo(ethToWeth(asset), network)).map(addr => addr.address.toLowerCase());
|
|
31
|
-
|
|
32
|
-
const getBaseAssetPriceFunction = (asset: string) => {
|
|
33
|
-
switch (asset) {
|
|
34
|
-
case 'wstETH':
|
|
35
|
-
return getWstETHPrice;
|
|
36
|
-
case 'ETH':
|
|
37
|
-
return getEthPrice;
|
|
38
|
-
default:
|
|
39
|
-
return getUSDCPrice;
|
|
40
|
-
}
|
|
41
|
-
};
|
|
42
|
-
|
|
43
|
-
export const _getCompoundV3MarketsData = async (provider: Client, network: NetworkNumber, selectedMarket: CompoundMarketData, defaultProvider: Client): Promise<CompoundV3MarketsData> => {
|
|
44
|
-
const contract = CompV3ViewContractViem(provider, network);
|
|
45
|
-
|
|
46
|
-
const [baseAssetPrice, compPrice, baseTokenInfo, collInfos] = await Promise.all([
|
|
47
|
-
getBaseAssetPriceFunction(selectedMarket.baseAsset)(defaultProvider),
|
|
48
|
-
getCompPrice(defaultProvider),
|
|
49
|
-
contract.read.getFullBaseTokenInfo([selectedMarket.baseMarketAddress]),
|
|
50
|
-
contract.read.getFullCollInfos([selectedMarket.baseMarketAddress]),
|
|
51
|
-
]);
|
|
52
|
-
|
|
53
|
-
const supportedAssetsAddresses = getSupportedAssetsAddressesForMarket(selectedMarket, network);
|
|
54
|
-
|
|
55
|
-
const colls = collInfos
|
|
56
|
-
.filter((coll: any) => supportedAssetsAddresses.includes(coll.tokenAddr.toLowerCase()))
|
|
57
|
-
.map((coll: any) => formatMarketData(coll, network, baseAssetPrice)) as CompoundV3AssetData[];
|
|
58
|
-
|
|
59
|
-
for (const coll of colls) {
|
|
60
|
-
if (STAKING_ASSETS.includes(coll.symbol)) {
|
|
61
|
-
coll.supplyIncentives.push({
|
|
62
|
-
apy: await getStakingApy(coll.symbol),
|
|
63
|
-
token: coll.symbol,
|
|
64
|
-
incentiveKind: IncentiveKind.Staking,
|
|
65
|
-
description: `Native ${coll.symbol} yield.`,
|
|
66
|
-
});
|
|
67
|
-
}
|
|
68
|
-
}
|
|
69
|
-
const base = formatBaseData(baseTokenInfo, network, baseAssetPrice);
|
|
70
|
-
|
|
71
|
-
const payload: CompoundV3AssetsData = {};
|
|
72
|
-
|
|
73
|
-
const baseObj = { ...base, ...(await getIncentiveApys(base, compPrice)) };
|
|
74
|
-
const allAssets = [baseObj, ...colls];
|
|
75
|
-
|
|
76
|
-
allAssets
|
|
77
|
-
.sort((a, b) => {
|
|
78
|
-
const aMarket = new Dec(a.price).times(a.totalSupply).toString();
|
|
79
|
-
const bMarket = new Dec(b.price).times(b.totalSupply).toString();
|
|
80
|
-
|
|
81
|
-
return new Dec(bMarket).minus(aMarket).toNumber();
|
|
82
|
-
})
|
|
83
|
-
.forEach((market, i) => {
|
|
84
|
-
payload[market.symbol] = { ...market, sortIndex: i };
|
|
85
|
-
});
|
|
86
|
-
|
|
87
|
-
return { assetsData: payload };
|
|
88
|
-
};
|
|
89
|
-
|
|
90
|
-
export const getCompoundV3MarketsData = async (provider: EthereumProvider, network: NetworkNumber, selectedMarket: CompoundMarketData, defaultProvider: EthereumProvider): Promise<CompoundV3MarketsData> => _getCompoundV3MarketsData(getViemProvider(provider, network), network, selectedMarket, getViemProvider(defaultProvider, network));
|
|
91
|
-
|
|
92
|
-
export const EMPTY_COMPOUND_V3_DATA = {
|
|
93
|
-
usedAssets: {},
|
|
94
|
-
suppliedUsd: '0',
|
|
95
|
-
borrowedUsd: '0',
|
|
96
|
-
borrowLimitUsd: '0',
|
|
97
|
-
leftToBorrowUsd: '0',
|
|
98
|
-
ratio: '0',
|
|
99
|
-
minRatio: '0',
|
|
100
|
-
netApy: '0',
|
|
101
|
-
incentiveUsd: '0',
|
|
102
|
-
totalInterestUsd: '0',
|
|
103
|
-
isSubscribedToAutomation: false,
|
|
104
|
-
automationResubscribeRequired: false,
|
|
105
|
-
isAllowed: false,
|
|
106
|
-
lastUpdated: Date.now(),
|
|
107
|
-
};
|
|
108
|
-
|
|
109
|
-
export const EMPTY_USED_ASSET = {
|
|
110
|
-
isSupplied: false,
|
|
111
|
-
isBorrowed: false,
|
|
112
|
-
supplied: '0',
|
|
113
|
-
suppliedUsd: '0',
|
|
114
|
-
borrowed: '0',
|
|
115
|
-
borrowedUsd: '0',
|
|
116
|
-
symbol: '',
|
|
117
|
-
collateral: true,
|
|
118
|
-
debt: '0',
|
|
119
|
-
};
|
|
120
|
-
|
|
121
|
-
export const _getCompoundV3AccountBalances = async (provider: Client, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress, marketAddress: EthAddress): Promise<PositionBalances> => {
|
|
122
|
-
let balances: PositionBalances = {
|
|
123
|
-
collateral: {},
|
|
124
|
-
debt: {},
|
|
125
|
-
};
|
|
126
|
-
|
|
127
|
-
if (!address) {
|
|
128
|
-
return balances;
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
const market = ({
|
|
132
|
-
[COMPOUND_V3_ETH(network).baseMarketAddress.toLowerCase()]: COMPOUND_V3_ETH(network),
|
|
133
|
-
[COMPOUND_V3_USDC(network).baseMarketAddress.toLowerCase()]: COMPOUND_V3_USDC(network),
|
|
134
|
-
[COMPOUND_V3_USDBC(network).baseMarketAddress.toLowerCase()]: COMPOUND_V3_USDBC(network),
|
|
135
|
-
[COMPOUND_V3_USDT(network).baseMarketAddress.toLowerCase()]: COMPOUND_V3_USDT(network),
|
|
136
|
-
[COMPOUND_V3_USDCe(network).baseMarketAddress.toLowerCase()]: COMPOUND_V3_USDCe(network),
|
|
137
|
-
})[marketAddress.toLowerCase()];
|
|
138
|
-
|
|
139
|
-
const loanInfoContract = CompV3ViewContractViem(provider, network, block);
|
|
140
|
-
const loanInfo = await loanInfoContract.read.getLoanData([market.baseMarketAddress, address], setViemBlockNumber(block));
|
|
141
|
-
const baseAssetInfo = getAssetInfo(wethToEth(market.baseAsset), network);
|
|
142
|
-
|
|
143
|
-
balances = {
|
|
144
|
-
collateral: {
|
|
145
|
-
[addressMapping ? baseAssetInfo.address.toLowerCase() : baseAssetInfo.symbol]: loanInfo.depositAmount.toString(),
|
|
146
|
-
},
|
|
147
|
-
debt: {
|
|
148
|
-
[addressMapping ? baseAssetInfo.address.toLowerCase() : baseAssetInfo.symbol]: loanInfo.borrowAmount.toString(),
|
|
149
|
-
},
|
|
150
|
-
};
|
|
151
|
-
|
|
152
|
-
loanInfo.collAddr.forEach((coll: string, i: number): void => {
|
|
153
|
-
const symbol = wethToEth(getAssetInfoByAddress(coll, network).symbol);
|
|
154
|
-
balances = {
|
|
155
|
-
...balances,
|
|
156
|
-
collateral: {
|
|
157
|
-
...balances.collateral,
|
|
158
|
-
[addressMapping ? getAssetInfo(symbol, network).address.toLowerCase() : symbol]: loanInfo.collAmounts[i].toString(),
|
|
159
|
-
},
|
|
160
|
-
};
|
|
161
|
-
});
|
|
162
|
-
|
|
163
|
-
return balances;
|
|
164
|
-
};
|
|
165
|
-
|
|
166
|
-
export const getCompoundV3AccountBalances = async (provider: EthereumProvider, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress, marketAddress: EthAddress): Promise<PositionBalances> => _getCompoundV3AccountBalances(getViemProvider(provider, network), network, block, addressMapping, address, marketAddress);
|
|
167
|
-
|
|
168
|
-
export const _getCompoundV3AccountData = async (
|
|
169
|
-
provider: Client,
|
|
170
|
-
network: NetworkNumber,
|
|
171
|
-
address: EthAddress,
|
|
172
|
-
proxyAddress: EthAddress,
|
|
173
|
-
extractedState: ({
|
|
174
|
-
selectedMarket: CompoundMarketData,
|
|
175
|
-
assetsData: CompoundV3AssetsData,
|
|
176
|
-
}),
|
|
177
|
-
): Promise<CompoundV3PositionData> => {
|
|
178
|
-
if (!address) throw new Error('No address provided');
|
|
179
|
-
const {
|
|
180
|
-
selectedMarket, assetsData,
|
|
181
|
-
} = extractedState;
|
|
182
|
-
|
|
183
|
-
let payload = {
|
|
184
|
-
...EMPTY_COMPOUND_V3_DATA,
|
|
185
|
-
lastUpdated: Date.now(),
|
|
186
|
-
};
|
|
187
|
-
|
|
188
|
-
const contract = CompV3ViewContractViem(provider, network);
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
const [loanData, isAllowed] = await Promise.all([
|
|
192
|
-
contract.read.getLoanData([selectedMarket.baseMarketAddress, address]),
|
|
193
|
-
contract.read.isAllowed([selectedMarket.baseMarketAddress, address, (proxyAddress || ZERO_ADDRESS)]),
|
|
194
|
-
]);
|
|
195
|
-
|
|
196
|
-
const usedAssets: CompoundV3UsedAssets = {};
|
|
197
|
-
|
|
198
|
-
const baseAssetInfo = getAssetInfo(selectedMarket.baseAsset);
|
|
199
|
-
const baseAssetSymbol = wethToEth(selectedMarket.baseAsset);
|
|
200
|
-
usedAssets[baseAssetSymbol] = { ...EMPTY_USED_ASSET, symbol: baseAssetSymbol, collateral: false };
|
|
201
|
-
if (loanData.depositAmount.toString() !== '0') {
|
|
202
|
-
usedAssets[baseAssetSymbol].isSupplied = true;
|
|
203
|
-
usedAssets[baseAssetSymbol].supplied = assetAmountInEth(loanData.depositAmount.toString(), baseAssetInfo.symbol);
|
|
204
|
-
usedAssets[baseAssetSymbol].suppliedUsd = new Dec(assetAmountInEth(loanData.depositValue.toString(), baseAssetInfo.symbol)).mul(assetsData[baseAssetSymbol].price).toString();
|
|
205
|
-
}
|
|
206
|
-
if (loanData.borrowAmount.toString() !== '0') {
|
|
207
|
-
usedAssets[baseAssetSymbol].isBorrowed = true;
|
|
208
|
-
usedAssets[baseAssetSymbol].borrowed = assetAmountInEth(loanData.borrowAmount.toString(), baseAssetInfo.symbol);
|
|
209
|
-
usedAssets[baseAssetSymbol].borrowedUsd = new Dec(
|
|
210
|
-
assetAmountInEth(loanData.borrowValue.toString(), baseAssetInfo.symbol),
|
|
211
|
-
)
|
|
212
|
-
.mul(assetsData[baseAssetSymbol].price)
|
|
213
|
-
.toString();
|
|
214
|
-
}
|
|
215
|
-
const supportedAssetsAddresses = getSupportedAssetsAddressesForMarket(selectedMarket, network);
|
|
216
|
-
|
|
217
|
-
loanData.collAddr.forEach((coll: string, i: number): void => {
|
|
218
|
-
// not filtering collAddr because there is no way of knowing how to filter loanData.collAmounts
|
|
219
|
-
if (!supportedAssetsAddresses.includes(coll.toLowerCase())) return;
|
|
220
|
-
const assetInfo = getAssetInfoByAddress(coll, network);
|
|
221
|
-
const symbol = wethToEth(assetInfo.symbol);
|
|
222
|
-
const supplied = assetAmountInEth(loanData.collAmounts[i].toString(), symbol);
|
|
223
|
-
const isSupplied = supplied !== '0';
|
|
224
|
-
const price = assetsData[symbol].price;
|
|
225
|
-
const suppliedUsd = new Dec(supplied).mul(price).toString();
|
|
226
|
-
usedAssets[symbol] = {
|
|
227
|
-
...usedAssets[symbol],
|
|
228
|
-
borrowed: '0',
|
|
229
|
-
borrowedUsd: '0',
|
|
230
|
-
isSupplied,
|
|
231
|
-
supplied,
|
|
232
|
-
suppliedUsd,
|
|
233
|
-
isBorrowed: false,
|
|
234
|
-
symbol,
|
|
235
|
-
collateral: true,
|
|
236
|
-
};
|
|
237
|
-
});
|
|
238
|
-
|
|
239
|
-
payload = {
|
|
240
|
-
...payload,
|
|
241
|
-
usedAssets,
|
|
242
|
-
...getCompoundV3AggregatedData({
|
|
243
|
-
usedAssets, assetsData, network, selectedMarket,
|
|
244
|
-
}),
|
|
245
|
-
isAllowed,
|
|
246
|
-
};
|
|
247
|
-
|
|
248
|
-
// Calculate borrow limits per asset
|
|
249
|
-
Object.values(payload.usedAssets).forEach((item: any) => {
|
|
250
|
-
if (item.isBorrowed) {
|
|
251
|
-
// eslint-disable-next-line no-param-reassign
|
|
252
|
-
item.limit = calculateBorrowingAssetLimit(item.borrowedUsd, payload.borrowLimitUsd);
|
|
253
|
-
}
|
|
254
|
-
});
|
|
255
|
-
|
|
256
|
-
return payload;
|
|
257
|
-
};
|
|
258
|
-
|
|
259
|
-
export const getCompoundV3AccountData = async (
|
|
260
|
-
provider: EthereumProvider,
|
|
261
|
-
network: NetworkNumber,
|
|
262
|
-
address: EthAddress,
|
|
263
|
-
proxyAddress: EthAddress,
|
|
264
|
-
extractedState: ({
|
|
265
|
-
selectedMarket: CompoundMarketData,
|
|
266
|
-
assetsData: CompoundV3AssetsData,
|
|
267
|
-
}),
|
|
268
|
-
): Promise<CompoundV3PositionData> => _getCompoundV3AccountData(getViemProvider(provider, network), network, address, proxyAddress, extractedState);
|
|
269
|
-
|
|
270
|
-
export const getCompoundV3FullPositionData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress, proxyAddress: EthAddress, selectedMarket: CompoundMarketData, defaultProvider: EthereumProvider): Promise<CompoundV3PositionData> => {
|
|
271
|
-
const marketData = await getCompoundV3MarketsData(provider, network, selectedMarket, defaultProvider);
|
|
272
|
-
const positionData = await getCompoundV3AccountData(provider, network, address, proxyAddress, { selectedMarket, assetsData: marketData.assetsData });
|
|
273
|
-
return positionData;
|
|
274
|
-
};
|
|
1
|
+
import Dec from 'decimal.js';
|
|
2
|
+
import {
|
|
3
|
+
assetAmountInEth, getAssetInfo, getAssetInfoByAddress,
|
|
4
|
+
} from '@defisaver/tokens';
|
|
5
|
+
import { Client } from 'viem';
|
|
6
|
+
import { CompV3ViewContractViem } from '../contracts';
|
|
7
|
+
import {
|
|
8
|
+
CompoundV3AssetData, CompoundMarketData, CompoundV3AssetsData, CompoundV3UsedAssets, CompoundV3MarketsData, CompoundV3PositionData,
|
|
9
|
+
} from '../types';
|
|
10
|
+
import {
|
|
11
|
+
Blockish, EthAddress, EthereumProvider, IncentiveKind, NetworkNumber, PositionBalances,
|
|
12
|
+
} from '../types/common';
|
|
13
|
+
import {
|
|
14
|
+
getStakingApy, STAKING_ASSETS,
|
|
15
|
+
} from '../staking';
|
|
16
|
+
import { ethToWeth, wethToEth } from '../services/utils';
|
|
17
|
+
import { ZERO_ADDRESS } from '../constants';
|
|
18
|
+
import { calculateBorrowingAssetLimit } from '../moneymarket';
|
|
19
|
+
import {
|
|
20
|
+
formatBaseData, formatMarketData, getCompoundV3AggregatedData, getIncentiveApys,
|
|
21
|
+
} from '../helpers/compoundHelpers';
|
|
22
|
+
import {
|
|
23
|
+
COMPOUND_V3_ETH, COMPOUND_V3_USDBC, COMPOUND_V3_USDC, COMPOUND_V3_USDCe, COMPOUND_V3_USDT,
|
|
24
|
+
} from '../markets/compound';
|
|
25
|
+
import {
|
|
26
|
+
getEthPrice, getCompPrice, getUSDCPrice, getWstETHPrice,
|
|
27
|
+
} from '../services/priceService';
|
|
28
|
+
import { getViemProvider, setViemBlockNumber } from '../services/viem';
|
|
29
|
+
|
|
30
|
+
const getSupportedAssetsAddressesForMarket = (selectedMarket: CompoundMarketData, network: NetworkNumber) => selectedMarket.collAssets.map(asset => getAssetInfo(ethToWeth(asset), network)).map(addr => addr.address.toLowerCase());
|
|
31
|
+
|
|
32
|
+
const getBaseAssetPriceFunction = (asset: string) => {
|
|
33
|
+
switch (asset) {
|
|
34
|
+
case 'wstETH':
|
|
35
|
+
return getWstETHPrice;
|
|
36
|
+
case 'ETH':
|
|
37
|
+
return getEthPrice;
|
|
38
|
+
default:
|
|
39
|
+
return getUSDCPrice;
|
|
40
|
+
}
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
export const _getCompoundV3MarketsData = async (provider: Client, network: NetworkNumber, selectedMarket: CompoundMarketData, defaultProvider: Client): Promise<CompoundV3MarketsData> => {
|
|
44
|
+
const contract = CompV3ViewContractViem(provider, network);
|
|
45
|
+
|
|
46
|
+
const [baseAssetPrice, compPrice, baseTokenInfo, collInfos] = await Promise.all([
|
|
47
|
+
getBaseAssetPriceFunction(selectedMarket.baseAsset)(defaultProvider),
|
|
48
|
+
getCompPrice(defaultProvider),
|
|
49
|
+
contract.read.getFullBaseTokenInfo([selectedMarket.baseMarketAddress]),
|
|
50
|
+
contract.read.getFullCollInfos([selectedMarket.baseMarketAddress]),
|
|
51
|
+
]);
|
|
52
|
+
|
|
53
|
+
const supportedAssetsAddresses = getSupportedAssetsAddressesForMarket(selectedMarket, network);
|
|
54
|
+
|
|
55
|
+
const colls = collInfos
|
|
56
|
+
.filter((coll: any) => supportedAssetsAddresses.includes(coll.tokenAddr.toLowerCase()))
|
|
57
|
+
.map((coll: any) => formatMarketData(coll, network, baseAssetPrice)) as CompoundV3AssetData[];
|
|
58
|
+
|
|
59
|
+
for (const coll of colls) {
|
|
60
|
+
if (STAKING_ASSETS.includes(coll.symbol)) {
|
|
61
|
+
coll.supplyIncentives.push({
|
|
62
|
+
apy: await getStakingApy(coll.symbol),
|
|
63
|
+
token: coll.symbol,
|
|
64
|
+
incentiveKind: IncentiveKind.Staking,
|
|
65
|
+
description: `Native ${coll.symbol} yield.`,
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
const base = formatBaseData(baseTokenInfo, network, baseAssetPrice);
|
|
70
|
+
|
|
71
|
+
const payload: CompoundV3AssetsData = {};
|
|
72
|
+
|
|
73
|
+
const baseObj = { ...base, ...(await getIncentiveApys(base, compPrice)) };
|
|
74
|
+
const allAssets = [baseObj, ...colls];
|
|
75
|
+
|
|
76
|
+
allAssets
|
|
77
|
+
.sort((a, b) => {
|
|
78
|
+
const aMarket = new Dec(a.price).times(a.totalSupply).toString();
|
|
79
|
+
const bMarket = new Dec(b.price).times(b.totalSupply).toString();
|
|
80
|
+
|
|
81
|
+
return new Dec(bMarket).minus(aMarket).toNumber();
|
|
82
|
+
})
|
|
83
|
+
.forEach((market, i) => {
|
|
84
|
+
payload[market.symbol] = { ...market, sortIndex: i };
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
return { assetsData: payload };
|
|
88
|
+
};
|
|
89
|
+
|
|
90
|
+
export const getCompoundV3MarketsData = async (provider: EthereumProvider, network: NetworkNumber, selectedMarket: CompoundMarketData, defaultProvider: EthereumProvider): Promise<CompoundV3MarketsData> => _getCompoundV3MarketsData(getViemProvider(provider, network), network, selectedMarket, getViemProvider(defaultProvider, network));
|
|
91
|
+
|
|
92
|
+
export const EMPTY_COMPOUND_V3_DATA = {
|
|
93
|
+
usedAssets: {},
|
|
94
|
+
suppliedUsd: '0',
|
|
95
|
+
borrowedUsd: '0',
|
|
96
|
+
borrowLimitUsd: '0',
|
|
97
|
+
leftToBorrowUsd: '0',
|
|
98
|
+
ratio: '0',
|
|
99
|
+
minRatio: '0',
|
|
100
|
+
netApy: '0',
|
|
101
|
+
incentiveUsd: '0',
|
|
102
|
+
totalInterestUsd: '0',
|
|
103
|
+
isSubscribedToAutomation: false,
|
|
104
|
+
automationResubscribeRequired: false,
|
|
105
|
+
isAllowed: false,
|
|
106
|
+
lastUpdated: Date.now(),
|
|
107
|
+
};
|
|
108
|
+
|
|
109
|
+
export const EMPTY_USED_ASSET = {
|
|
110
|
+
isSupplied: false,
|
|
111
|
+
isBorrowed: false,
|
|
112
|
+
supplied: '0',
|
|
113
|
+
suppliedUsd: '0',
|
|
114
|
+
borrowed: '0',
|
|
115
|
+
borrowedUsd: '0',
|
|
116
|
+
symbol: '',
|
|
117
|
+
collateral: true,
|
|
118
|
+
debt: '0',
|
|
119
|
+
};
|
|
120
|
+
|
|
121
|
+
export const _getCompoundV3AccountBalances = async (provider: Client, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress, marketAddress: EthAddress): Promise<PositionBalances> => {
|
|
122
|
+
let balances: PositionBalances = {
|
|
123
|
+
collateral: {},
|
|
124
|
+
debt: {},
|
|
125
|
+
};
|
|
126
|
+
|
|
127
|
+
if (!address) {
|
|
128
|
+
return balances;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
const market = ({
|
|
132
|
+
[COMPOUND_V3_ETH(network).baseMarketAddress.toLowerCase()]: COMPOUND_V3_ETH(network),
|
|
133
|
+
[COMPOUND_V3_USDC(network).baseMarketAddress.toLowerCase()]: COMPOUND_V3_USDC(network),
|
|
134
|
+
[COMPOUND_V3_USDBC(network).baseMarketAddress.toLowerCase()]: COMPOUND_V3_USDBC(network),
|
|
135
|
+
[COMPOUND_V3_USDT(network).baseMarketAddress.toLowerCase()]: COMPOUND_V3_USDT(network),
|
|
136
|
+
[COMPOUND_V3_USDCe(network).baseMarketAddress.toLowerCase()]: COMPOUND_V3_USDCe(network),
|
|
137
|
+
})[marketAddress.toLowerCase()];
|
|
138
|
+
|
|
139
|
+
const loanInfoContract = CompV3ViewContractViem(provider, network, block);
|
|
140
|
+
const loanInfo = await loanInfoContract.read.getLoanData([market.baseMarketAddress, address], setViemBlockNumber(block));
|
|
141
|
+
const baseAssetInfo = getAssetInfo(wethToEth(market.baseAsset), network);
|
|
142
|
+
|
|
143
|
+
balances = {
|
|
144
|
+
collateral: {
|
|
145
|
+
[addressMapping ? baseAssetInfo.address.toLowerCase() : baseAssetInfo.symbol]: loanInfo.depositAmount.toString(),
|
|
146
|
+
},
|
|
147
|
+
debt: {
|
|
148
|
+
[addressMapping ? baseAssetInfo.address.toLowerCase() : baseAssetInfo.symbol]: loanInfo.borrowAmount.toString(),
|
|
149
|
+
},
|
|
150
|
+
};
|
|
151
|
+
|
|
152
|
+
loanInfo.collAddr.forEach((coll: string, i: number): void => {
|
|
153
|
+
const symbol = wethToEth(getAssetInfoByAddress(coll, network).symbol);
|
|
154
|
+
balances = {
|
|
155
|
+
...balances,
|
|
156
|
+
collateral: {
|
|
157
|
+
...balances.collateral,
|
|
158
|
+
[addressMapping ? getAssetInfo(symbol, network).address.toLowerCase() : symbol]: loanInfo.collAmounts[i].toString(),
|
|
159
|
+
},
|
|
160
|
+
};
|
|
161
|
+
});
|
|
162
|
+
|
|
163
|
+
return balances;
|
|
164
|
+
};
|
|
165
|
+
|
|
166
|
+
export const getCompoundV3AccountBalances = async (provider: EthereumProvider, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress, marketAddress: EthAddress): Promise<PositionBalances> => _getCompoundV3AccountBalances(getViemProvider(provider, network), network, block, addressMapping, address, marketAddress);
|
|
167
|
+
|
|
168
|
+
export const _getCompoundV3AccountData = async (
|
|
169
|
+
provider: Client,
|
|
170
|
+
network: NetworkNumber,
|
|
171
|
+
address: EthAddress,
|
|
172
|
+
proxyAddress: EthAddress,
|
|
173
|
+
extractedState: ({
|
|
174
|
+
selectedMarket: CompoundMarketData,
|
|
175
|
+
assetsData: CompoundV3AssetsData,
|
|
176
|
+
}),
|
|
177
|
+
): Promise<CompoundV3PositionData> => {
|
|
178
|
+
if (!address) throw new Error('No address provided');
|
|
179
|
+
const {
|
|
180
|
+
selectedMarket, assetsData,
|
|
181
|
+
} = extractedState;
|
|
182
|
+
|
|
183
|
+
let payload = {
|
|
184
|
+
...EMPTY_COMPOUND_V3_DATA,
|
|
185
|
+
lastUpdated: Date.now(),
|
|
186
|
+
};
|
|
187
|
+
|
|
188
|
+
const contract = CompV3ViewContractViem(provider, network);
|
|
189
|
+
|
|
190
|
+
|
|
191
|
+
const [loanData, isAllowed] = await Promise.all([
|
|
192
|
+
contract.read.getLoanData([selectedMarket.baseMarketAddress, address]),
|
|
193
|
+
contract.read.isAllowed([selectedMarket.baseMarketAddress, address, (proxyAddress || ZERO_ADDRESS)]),
|
|
194
|
+
]);
|
|
195
|
+
|
|
196
|
+
const usedAssets: CompoundV3UsedAssets = {};
|
|
197
|
+
|
|
198
|
+
const baseAssetInfo = getAssetInfo(selectedMarket.baseAsset);
|
|
199
|
+
const baseAssetSymbol = wethToEth(selectedMarket.baseAsset);
|
|
200
|
+
usedAssets[baseAssetSymbol] = { ...EMPTY_USED_ASSET, symbol: baseAssetSymbol, collateral: false };
|
|
201
|
+
if (loanData.depositAmount.toString() !== '0') {
|
|
202
|
+
usedAssets[baseAssetSymbol].isSupplied = true;
|
|
203
|
+
usedAssets[baseAssetSymbol].supplied = assetAmountInEth(loanData.depositAmount.toString(), baseAssetInfo.symbol);
|
|
204
|
+
usedAssets[baseAssetSymbol].suppliedUsd = new Dec(assetAmountInEth(loanData.depositValue.toString(), baseAssetInfo.symbol)).mul(assetsData[baseAssetSymbol].price).toString();
|
|
205
|
+
}
|
|
206
|
+
if (loanData.borrowAmount.toString() !== '0') {
|
|
207
|
+
usedAssets[baseAssetSymbol].isBorrowed = true;
|
|
208
|
+
usedAssets[baseAssetSymbol].borrowed = assetAmountInEth(loanData.borrowAmount.toString(), baseAssetInfo.symbol);
|
|
209
|
+
usedAssets[baseAssetSymbol].borrowedUsd = new Dec(
|
|
210
|
+
assetAmountInEth(loanData.borrowValue.toString(), baseAssetInfo.symbol),
|
|
211
|
+
)
|
|
212
|
+
.mul(assetsData[baseAssetSymbol].price)
|
|
213
|
+
.toString();
|
|
214
|
+
}
|
|
215
|
+
const supportedAssetsAddresses = getSupportedAssetsAddressesForMarket(selectedMarket, network);
|
|
216
|
+
|
|
217
|
+
loanData.collAddr.forEach((coll: string, i: number): void => {
|
|
218
|
+
// not filtering collAddr because there is no way of knowing how to filter loanData.collAmounts
|
|
219
|
+
if (!supportedAssetsAddresses.includes(coll.toLowerCase())) return;
|
|
220
|
+
const assetInfo = getAssetInfoByAddress(coll, network);
|
|
221
|
+
const symbol = wethToEth(assetInfo.symbol);
|
|
222
|
+
const supplied = assetAmountInEth(loanData.collAmounts[i].toString(), symbol);
|
|
223
|
+
const isSupplied = supplied !== '0';
|
|
224
|
+
const price = assetsData[symbol].price;
|
|
225
|
+
const suppliedUsd = new Dec(supplied).mul(price).toString();
|
|
226
|
+
usedAssets[symbol] = {
|
|
227
|
+
...usedAssets[symbol],
|
|
228
|
+
borrowed: '0',
|
|
229
|
+
borrowedUsd: '0',
|
|
230
|
+
isSupplied,
|
|
231
|
+
supplied,
|
|
232
|
+
suppliedUsd,
|
|
233
|
+
isBorrowed: false,
|
|
234
|
+
symbol,
|
|
235
|
+
collateral: true,
|
|
236
|
+
};
|
|
237
|
+
});
|
|
238
|
+
|
|
239
|
+
payload = {
|
|
240
|
+
...payload,
|
|
241
|
+
usedAssets,
|
|
242
|
+
...getCompoundV3AggregatedData({
|
|
243
|
+
usedAssets, assetsData, network, selectedMarket,
|
|
244
|
+
}),
|
|
245
|
+
isAllowed,
|
|
246
|
+
};
|
|
247
|
+
|
|
248
|
+
// Calculate borrow limits per asset
|
|
249
|
+
Object.values(payload.usedAssets).forEach((item: any) => {
|
|
250
|
+
if (item.isBorrowed) {
|
|
251
|
+
// eslint-disable-next-line no-param-reassign
|
|
252
|
+
item.limit = calculateBorrowingAssetLimit(item.borrowedUsd, payload.borrowLimitUsd);
|
|
253
|
+
}
|
|
254
|
+
});
|
|
255
|
+
|
|
256
|
+
return payload;
|
|
257
|
+
};
|
|
258
|
+
|
|
259
|
+
export const getCompoundV3AccountData = async (
|
|
260
|
+
provider: EthereumProvider,
|
|
261
|
+
network: NetworkNumber,
|
|
262
|
+
address: EthAddress,
|
|
263
|
+
proxyAddress: EthAddress,
|
|
264
|
+
extractedState: ({
|
|
265
|
+
selectedMarket: CompoundMarketData,
|
|
266
|
+
assetsData: CompoundV3AssetsData,
|
|
267
|
+
}),
|
|
268
|
+
): Promise<CompoundV3PositionData> => _getCompoundV3AccountData(getViemProvider(provider, network), network, address, proxyAddress, extractedState);
|
|
269
|
+
|
|
270
|
+
export const getCompoundV3FullPositionData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress, proxyAddress: EthAddress, selectedMarket: CompoundMarketData, defaultProvider: EthereumProvider): Promise<CompoundV3PositionData> => {
|
|
271
|
+
const marketData = await getCompoundV3MarketsData(provider, network, selectedMarket, defaultProvider);
|
|
272
|
+
const positionData = await getCompoundV3AccountData(provider, network, address, proxyAddress, { selectedMarket, assetsData: marketData.assetsData });
|
|
273
|
+
return positionData;
|
|
274
|
+
};
|