@defisaver/positions-sdk 2.1.8 → 2.1.9

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.
Files changed (96) hide show
  1. package/.mocharc.json +4 -4
  2. package/.nvmrc +1 -1
  3. package/CLAUDE.md +32 -0
  4. package/README.md +64 -64
  5. package/cjs/fluid/index.js +40 -12
  6. package/cjs/helpers/morphoBlueHelpers/index.js +66 -66
  7. package/cjs/markets/fluid/index.d.ts +4 -0
  8. package/cjs/markets/fluid/index.js +4 -0
  9. package/cjs/types/fluid.d.ts +7 -3
  10. package/cjs/types/fluid.js +4 -0
  11. package/esm/fluid/index.js +40 -12
  12. package/esm/helpers/morphoBlueHelpers/index.js +66 -66
  13. package/esm/markets/fluid/index.d.ts +4 -0
  14. package/esm/markets/fluid/index.js +4 -0
  15. package/esm/types/fluid.d.ts +7 -3
  16. package/esm/types/fluid.js +4 -0
  17. package/package.json +47 -47
  18. package/src/aaveV2/index.ts +240 -240
  19. package/src/aaveV3/index.ts +614 -614
  20. package/src/aaveV3/merit.ts +97 -97
  21. package/src/aaveV3/merkl.ts +74 -74
  22. package/src/claiming/aaveV3.ts +154 -154
  23. package/src/claiming/compV3.ts +22 -22
  24. package/src/claiming/index.ts +12 -12
  25. package/src/claiming/king.ts +66 -66
  26. package/src/claiming/morphoBlue.ts +118 -118
  27. package/src/claiming/spark.ts +225 -225
  28. package/src/compoundV2/index.ts +244 -244
  29. package/src/compoundV3/index.ts +274 -274
  30. package/src/config/contracts.ts +1251 -1251
  31. package/src/constants/index.ts +10 -10
  32. package/src/contracts.ts +120 -120
  33. package/src/curveUsd/index.ts +254 -254
  34. package/src/eulerV2/index.ts +324 -324
  35. package/src/exchange/index.ts +25 -25
  36. package/src/fluid/index.ts +1668 -1638
  37. package/src/helpers/aaveHelpers/index.ts +187 -187
  38. package/src/helpers/compoundHelpers/index.ts +283 -283
  39. package/src/helpers/curveUsdHelpers/index.ts +40 -40
  40. package/src/helpers/eulerHelpers/index.ts +222 -222
  41. package/src/helpers/fluidHelpers/index.ts +326 -326
  42. package/src/helpers/index.ts +10 -10
  43. package/src/helpers/liquityV2Helpers/index.ts +82 -82
  44. package/src/helpers/llamaLendHelpers/index.ts +53 -53
  45. package/src/helpers/makerHelpers/index.ts +52 -52
  46. package/src/helpers/morphoBlueHelpers/index.ts +396 -396
  47. package/src/helpers/sparkHelpers/index.ts +155 -155
  48. package/src/index.ts +47 -47
  49. package/src/liquity/index.ts +159 -159
  50. package/src/liquityV2/index.ts +657 -657
  51. package/src/llamaLend/index.ts +305 -305
  52. package/src/maker/index.ts +223 -223
  53. package/src/markets/aave/index.ts +116 -116
  54. package/src/markets/aave/marketAssets.ts +54 -54
  55. package/src/markets/compound/index.ts +238 -238
  56. package/src/markets/compound/marketsAssets.ts +97 -97
  57. package/src/markets/curveUsd/index.ts +69 -69
  58. package/src/markets/euler/index.ts +26 -26
  59. package/src/markets/fluid/index.ts +2460 -2456
  60. package/src/markets/index.ts +25 -25
  61. package/src/markets/liquityV2/index.ts +102 -102
  62. package/src/markets/llamaLend/contractAddresses.ts +141 -141
  63. package/src/markets/llamaLend/index.ts +235 -235
  64. package/src/markets/morphoBlue/index.ts +895 -895
  65. package/src/markets/spark/index.ts +29 -29
  66. package/src/markets/spark/marketAssets.ts +12 -12
  67. package/src/moneymarket/moneymarketCommonService.ts +80 -80
  68. package/src/morphoBlue/index.ts +274 -274
  69. package/src/portfolio/index.ts +570 -570
  70. package/src/services/priceService.ts +159 -159
  71. package/src/services/utils.ts +115 -115
  72. package/src/services/viem.ts +34 -34
  73. package/src/setup.ts +8 -8
  74. package/src/spark/index.ts +445 -445
  75. package/src/staking/eligibility.ts +53 -53
  76. package/src/staking/index.ts +1 -1
  77. package/src/staking/staking.ts +170 -170
  78. package/src/types/aave.ts +189 -189
  79. package/src/types/claiming.ts +109 -109
  80. package/src/types/common.ts +107 -107
  81. package/src/types/compound.ts +136 -136
  82. package/src/types/curveUsd.ts +123 -123
  83. package/src/types/euler.ts +175 -175
  84. package/src/types/fluid.ts +452 -448
  85. package/src/types/index.ts +13 -13
  86. package/src/types/liquity.ts +30 -30
  87. package/src/types/liquityV2.ts +126 -126
  88. package/src/types/llamaLend.ts +159 -159
  89. package/src/types/maker.ts +63 -63
  90. package/src/types/merit.ts +1 -1
  91. package/src/types/merkl.ts +70 -70
  92. package/src/types/morphoBlue.ts +194 -194
  93. package/src/types/portfolio.ts +60 -60
  94. package/src/types/spark.ts +135 -135
  95. package/src/umbrella/index.ts +69 -69
  96. package/src/umbrella/umbrellaUtils.ts +29 -29
@@ -1,305 +1,305 @@
1
- import Dec from 'decimal.js';
2
- import { assetAmountInEth, getAssetInfo } from '@defisaver/tokens';
3
- import { Client } from 'viem';
4
- import {
5
- LlamaLendAssetsData,
6
- LlamaLendGlobalMarketData, LlamaLendMarketData, LlamaLendStatus, LlamaLendUsedAssets, LlamaLendUserData,
7
- } from '../types';
8
- import {
9
- Blockish, EthAddress, EthereumProvider, IncentiveKind, NetworkNumber, PositionBalances,
10
- } from '../types/common';
11
- import { LlamaLendViewContractViem } from '../contracts';
12
- import { getLlamaLendAggregatedData } from '../helpers/llamaLendHelpers';
13
- import { getEthAmountForDecimals, wethToEth } from '../services/utils';
14
- import { getLlamaLendMarketFromControllerAddress } from '../markets/llamaLend';
15
- import { getStakingApy, STAKING_ASSETS } from '../staking';
16
- import { getViemProvider, setViemBlockNumber } from '../services/viem';
17
-
18
- const getAndFormatBands = async (provider: Client, network: NetworkNumber, selectedMarket: LlamaLendMarketData, _minBand: string, _maxBand: string) => {
19
- const contract = LlamaLendViewContractViem(provider, network);
20
- const minBand = parseInt(_minBand, 10);
21
- const maxBand = parseInt(_maxBand, 10);
22
- const pivots: number[] = [];
23
-
24
- // getBandsData uses a lot of gas to get all of the bands at once, so we use pagination and fetch 200 bands at a time
25
- let i = minBand;
26
- while (i < maxBand) {
27
- i += 200;
28
- if (i > maxBand) {
29
- pivots.push(maxBand);
30
- } else {
31
- pivots.push(i);
32
- }
33
- }
34
-
35
- const bandsData = (await Promise.all(pivots.map(async (pivot, index) => {
36
- let start = 0;
37
- if (index === 0) {
38
- start = minBand;
39
- } else {
40
- start = pivots[index - 1] + 1;
41
- }
42
- const pivotedBandsData = await contract.read.getBandsData([selectedMarket.controllerAddress, BigInt(start), BigInt(pivot)]);
43
- return pivotedBandsData;
44
- }))).flat();
45
-
46
- return bandsData.map((band) => ({
47
- id: band.id.toString(),
48
- collAmount: assetAmountInEth(band.collAmount.toString()),
49
- debtAmount: assetAmountInEth(band.debtAmount.toString()),
50
- lowPrice: assetAmountInEth(band.lowPrice.toString()),
51
- highPrice: assetAmountInEth(band.highPrice.toString()),
52
- }));
53
- };
54
-
55
- export const _getLlamaLendGlobalData = async (provider: Client, network: NetworkNumber, selectedMarket: LlamaLendMarketData): Promise<LlamaLendGlobalMarketData> => {
56
- const contract = LlamaLendViewContractViem(provider, network);
57
-
58
- const collAsset = selectedMarket.collAsset;
59
- const debtAsset = selectedMarket.baseAsset;
60
-
61
- const data = await contract.read.globalData([selectedMarket.controllerAddress]);
62
-
63
- // all prices are in 18 decimals
64
- const oraclePrice = getEthAmountForDecimals(data.oraclePrice.toString(), 18);
65
- const collPriceUsd = collAsset === 'crvUSD' ? '1' : new Dec(1).mul(oraclePrice).toDP(18).toString();
66
- const debtPriceUsd = debtAsset === 'crvUSD' ? '1' : new Dec(1).div(oraclePrice).toDP(18).toString();
67
-
68
- const totalDebt = assetAmountInEth(data.totalDebt.toString(), debtAsset);
69
- const totalDebtSupplied = assetAmountInEth(data.debtTokenTotalSupply.toString(), debtAsset);
70
- const utilization = new Dec(totalDebtSupplied).gt(0)
71
- ? new Dec(totalDebt).div(totalDebtSupplied).mul(100).toString()
72
- : '0';
73
- const ammPrice = assetAmountInEth(data.ammPrice.toString(), debtAsset);
74
-
75
- const rate = assetAmountInEth(data.ammRate.toString());
76
- const futureRate = assetAmountInEth(data.monetaryPolicyRate.toString());
77
-
78
- const exponentRate = new Dec(rate).mul(365).mul(86400);
79
- const exponentFutureRate = new Dec(futureRate).mul(365).mul(86400);
80
- const borrowRate = new Dec(new Dec(2.718281828459).pow(exponentRate).minus(1)).mul(100)
81
- .toString();
82
- const futureBorrowRate = new Dec(new Dec(2.718281828459).pow(exponentFutureRate).minus(1)).mul(100)
83
- .toString();
84
-
85
- const bandsData = await getAndFormatBands(provider, network, selectedMarket, data.minBand.toString(), data.maxBand.toString());
86
- const cap = assetAmountInEth(data.debtTokenTotalSupply.toString(), debtAsset);
87
- const leftToBorrow = getEthAmountForDecimals(data.debtTokenLeftToBorrow.toString(), 18);
88
-
89
- const debtInAYearBN = new Dec(totalDebt).mul(new Dec(2.718281828459).pow(exponentRate).toNumber());
90
- const lendRate = debtInAYearBN.minus(totalDebt).div(cap).mul(100).toString();
91
-
92
- const assetsData: LlamaLendAssetsData = {};
93
- assetsData[debtAsset] = {
94
- symbol: debtAsset,
95
- address: data.debtToken,
96
- price: debtPriceUsd,
97
- supplyRate: lendRate,
98
- borrowRate,
99
- canBeSupplied: true,
100
- canBeBorrowed: true,
101
- supplyIncentives: [],
102
- borrowIncentives: [],
103
- };
104
-
105
- assetsData[collAsset] = {
106
- symbol: collAsset,
107
- address: data.collateralToken,
108
- price: collPriceUsd,
109
- supplyRate: '0',
110
- borrowRate: '0',
111
- canBeSupplied: true,
112
- canBeBorrowed: false,
113
- supplyIncentives: [],
114
- borrowIncentives: [],
115
- };
116
-
117
- if (STAKING_ASSETS.includes(collAsset)) {
118
- assetsData[collAsset].supplyIncentives.push({
119
- apy: await getStakingApy(collAsset),
120
- token: collAsset,
121
- incentiveKind: IncentiveKind.Staking,
122
- description: `Native ${collAsset} yield.`,
123
- });
124
- }
125
-
126
- return {
127
- A: data.A.toString(),
128
- loanDiscount: data.loanDiscount.toString(),
129
- activeBand: data.activeBand.toString(),
130
- monetaryPolicyRate: data.monetaryPolicyRate.toString(),
131
- ammRate: data.ammRate.toString(),
132
- minBand: data.minBand.toString(),
133
- maxBand: data.maxBand.toString(),
134
- assetsData,
135
- totalDebt,
136
- totalDebtSupplied,
137
- utilization,
138
- ammPrice,
139
- oraclePrice: assetAmountInEth(data.oraclePrice.toString(), debtAsset),
140
- basePrice: assetAmountInEth(data.basePrice.toString(), debtAsset),
141
- minted: assetAmountInEth(data.minted.toString(), debtAsset),
142
- redeemed: assetAmountInEth(data.redeemed.toString(), debtAsset),
143
- borrowRate,
144
- lendRate,
145
- futureBorrowRate,
146
- bands: bandsData,
147
- leftToBorrow,
148
- };
149
- };
150
-
151
- export const getLlamaLendGlobalData = async (
152
- provider: EthereumProvider,
153
- network: NetworkNumber,
154
- selectedMarket: LlamaLendMarketData,
155
- ): Promise<LlamaLendGlobalMarketData> => _getLlamaLendGlobalData(getViemProvider(provider, network), network, selectedMarket);
156
-
157
- const getStatusForUser = (bandRange: string[], activeBand: string, debtSupplied: string, collSupplied: string, healthPercent: string) => {
158
- // if bands are equal, that can only be [0,0] which means user doesn't have loan (min number of bands is 4)
159
- if (new Dec(bandRange[0]).eq(bandRange[1])) return LlamaLendStatus.Nonexistant;
160
- // if user doesn't have debtAsset as collateral, then his position is not in soft liquidation
161
- if (new Dec(debtSupplied).lte(0)) {
162
- const isHealthRisky = new Dec(healthPercent).lt(10);
163
- if (new Dec(bandRange[0]).minus(activeBand).lte(3) || isHealthRisky) return LlamaLendStatus.Risk; // if user band is less than 3 bands away from active band, his position is at risk
164
- return LlamaLendStatus.Safe;
165
- }
166
- if (new Dec(bandRange[0]).lte(activeBand) && new Dec(bandRange[1]).gte(activeBand)) return LlamaLendStatus.SoftLiquidating; // user has debtAsset as coll so he is in soft liquidation
167
- if (new Dec(collSupplied).lte(0) || new Dec(bandRange[1]).lte(activeBand)) return LlamaLendStatus.SoftLiquidated; // or is fully soft liquidated
168
- return LlamaLendStatus.Nonexistant;
169
- };
170
-
171
- export const _getLlamaLendAccountBalances = async (provider: Client, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress, controllerAddress: EthAddress): Promise<PositionBalances> => {
172
- let balances: PositionBalances = {
173
- collateral: {},
174
- debt: {},
175
- };
176
-
177
- if (!address) {
178
- return balances;
179
- }
180
-
181
- const contract = LlamaLendViewContractViem(provider, network, block);
182
-
183
- const selectedMarket = getLlamaLendMarketFromControllerAddress(controllerAddress, network);
184
- const data = await contract.read.userData([selectedMarket.controllerAddress, address], setViemBlockNumber(block));
185
-
186
- balances = {
187
- collateral: {
188
- [addressMapping ? getAssetInfo(wethToEth(selectedMarket.collAsset), network).address.toLowerCase() : wethToEth(selectedMarket.collAsset)]: data.marketCollateralAmount.toString(),
189
- },
190
- debt: {
191
- [addressMapping ? getAssetInfo(wethToEth(selectedMarket.baseAsset), network).address.toLowerCase() : wethToEth(selectedMarket.baseAsset)]: data.debtAmount.toString(),
192
- },
193
- };
194
-
195
- return balances;
196
- };
197
-
198
- export const getLlamaLendAccountBalances = async (
199
- provider: EthereumProvider,
200
- network: NetworkNumber,
201
- block: Blockish,
202
- addressMapping: boolean,
203
- address: EthAddress,
204
- controllerAddress: EthAddress,
205
- ): Promise<PositionBalances> => _getLlamaLendAccountBalances(getViemProvider(provider, network), network, block, addressMapping, address, controllerAddress);
206
-
207
- export const _getLlamaLendUserData = async (provider: Client, network: NetworkNumber, address: EthAddress, selectedMarket: LlamaLendMarketData, marketData: LlamaLendGlobalMarketData): Promise<LlamaLendUserData> => {
208
- const contract = LlamaLendViewContractViem(provider, network);
209
- const { assetsData } = marketData;
210
-
211
- const data = await contract.read.userData([selectedMarket.controllerAddress, address]);
212
- const collAsset = selectedMarket.collAsset;
213
- const debtAsset = selectedMarket.baseAsset;
214
-
215
- const collPrice = assetsData[collAsset].price;
216
- const debtPrice = assetsData[debtAsset].price;
217
-
218
- const health = assetAmountInEth(data.health.toString());
219
- const healthPercent = new Dec(health).mul(100).toString();
220
-
221
- const collSupplied = assetAmountInEth(data.marketCollateralAmount.toString(), collAsset);
222
- const collSuppliedUsd = new Dec(collSupplied).mul(collPrice).toString();
223
-
224
- const debtSupplied = assetAmountInEth(data.debtTokenCollateralAmount.toString(), debtAsset);
225
- const debtSuppliedUsd = new Dec(debtSupplied).mul(debtPrice).toString();
226
-
227
- const debtSuppliedForYield = assetAmountInEth(data.debtTokenSuppliedAssets.toString(), debtAsset);
228
- const debtSuppliedForYieldUsd = new Dec(debtSupplied).mul(debtPrice).toString();
229
-
230
- const debtBorrowed = assetAmountInEth(data.debtAmount.toString(), debtAsset);
231
- const debtBorrowedUsd = new Dec(debtBorrowed).mul(debtPrice).toString();
232
- const shares = assetAmountInEth(data.debtTokenSuppliedShares.toString(), debtAsset);
233
-
234
- const usedAssets: LlamaLendUsedAssets = {
235
- [collAsset]: {
236
- isSupplied: new Dec(collSupplied).gt('0'),
237
- supplied: collSupplied,
238
- suppliedUsd: collSuppliedUsd,
239
- borrowed: '0',
240
- borrowedUsd: '0',
241
- isBorrowed: false,
242
- symbol: collAsset,
243
- collateral: true,
244
- price: collPrice,
245
- },
246
- [debtAsset]: {
247
- isSupplied: new Dec(debtSupplied).gt('0') || new Dec(debtSuppliedForYield).gt('0'),
248
- collateral: new Dec(debtSupplied).gt('0'),
249
- supplied: debtSupplied,
250
- suppliedUsd: debtSuppliedUsd,
251
- suppliedForYield: debtSuppliedForYield,
252
- suppliedForYieldUsd: debtSuppliedForYieldUsd,
253
- borrowed: debtBorrowed,
254
- borrowedUsd: debtBorrowedUsd,
255
- isBorrowed: new Dec(debtBorrowed).gt('0'),
256
- symbol: debtAsset,
257
- price: debtPrice,
258
- shares,
259
- },
260
- };
261
-
262
- const priceHigh = assetAmountInEth(data.priceHigh.toString());
263
- const priceLow = assetAmountInEth(data.priceLow.toString());
264
-
265
- const _userBands = data.loanExists ? (await getAndFormatBands(provider, network, selectedMarket, data.bandRange[0].toString(), data.bandRange[1].toString())) : [];
266
-
267
- const status = data.loanExists ? getStatusForUser(data.bandRange.map(b => b.toString()), marketData.activeBand, debtSupplied, collSupplied, healthPercent) : LlamaLendStatus.Nonexistant;
268
-
269
- const userBands = _userBands.map((band, index) => ({
270
- ...band,
271
- userDebtAmount: assetAmountInEth(data.usersBands[0][index].toString(), debtAsset),
272
- userCollAmount: assetAmountInEth(data.usersBands[1][index].toString(), collAsset),
273
- })).sort((a, b) => parseInt(b.id, 10) - parseInt(a.id, 10));
274
-
275
- return {
276
- ...data,
277
- debtAmount: assetAmountInEth(data.debtAmount.toString(), debtAsset),
278
- health,
279
- healthPercent,
280
- priceHigh,
281
- priceLow,
282
- liquidationDiscount: assetAmountInEth(data.liquidationDiscount.toString()),
283
- numOfBands: data.N.toString(),
284
- usedAssets,
285
- status,
286
- ...getLlamaLendAggregatedData({
287
- loanExists: data.loanExists, usedAssets, network: NetworkNumber.Eth, selectedMarket, numOfBands: data.N.toString(), assetsData,
288
- }),
289
- userBands,
290
- };
291
- };
292
-
293
- export const getLlamaLendUserData = async (
294
- provider: EthereumProvider,
295
- network: NetworkNumber,
296
- address: EthAddress,
297
- selectedMarket: LlamaLendMarketData,
298
- marketData: LlamaLendGlobalMarketData,
299
- ): Promise<LlamaLendUserData> => _getLlamaLendUserData(getViemProvider(provider, network), network, address, selectedMarket, marketData);
300
-
301
- export const getLlamaLendFullPositionData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress, selectedMarket: LlamaLendMarketData): Promise<LlamaLendUserData> => {
302
- const marketData = await getLlamaLendGlobalData(provider, network, selectedMarket);
303
- const positionData = await getLlamaLendUserData(provider, network, address, selectedMarket, marketData);
304
- return positionData;
305
- };
1
+ import Dec from 'decimal.js';
2
+ import { assetAmountInEth, getAssetInfo } from '@defisaver/tokens';
3
+ import { Client } from 'viem';
4
+ import {
5
+ LlamaLendAssetsData,
6
+ LlamaLendGlobalMarketData, LlamaLendMarketData, LlamaLendStatus, LlamaLendUsedAssets, LlamaLendUserData,
7
+ } from '../types';
8
+ import {
9
+ Blockish, EthAddress, EthereumProvider, IncentiveKind, NetworkNumber, PositionBalances,
10
+ } from '../types/common';
11
+ import { LlamaLendViewContractViem } from '../contracts';
12
+ import { getLlamaLendAggregatedData } from '../helpers/llamaLendHelpers';
13
+ import { getEthAmountForDecimals, wethToEth } from '../services/utils';
14
+ import { getLlamaLendMarketFromControllerAddress } from '../markets/llamaLend';
15
+ import { getStakingApy, STAKING_ASSETS } from '../staking';
16
+ import { getViemProvider, setViemBlockNumber } from '../services/viem';
17
+
18
+ const getAndFormatBands = async (provider: Client, network: NetworkNumber, selectedMarket: LlamaLendMarketData, _minBand: string, _maxBand: string) => {
19
+ const contract = LlamaLendViewContractViem(provider, network);
20
+ const minBand = parseInt(_minBand, 10);
21
+ const maxBand = parseInt(_maxBand, 10);
22
+ const pivots: number[] = [];
23
+
24
+ // getBandsData uses a lot of gas to get all of the bands at once, so we use pagination and fetch 200 bands at a time
25
+ let i = minBand;
26
+ while (i < maxBand) {
27
+ i += 200;
28
+ if (i > maxBand) {
29
+ pivots.push(maxBand);
30
+ } else {
31
+ pivots.push(i);
32
+ }
33
+ }
34
+
35
+ const bandsData = (await Promise.all(pivots.map(async (pivot, index) => {
36
+ let start = 0;
37
+ if (index === 0) {
38
+ start = minBand;
39
+ } else {
40
+ start = pivots[index - 1] + 1;
41
+ }
42
+ const pivotedBandsData = await contract.read.getBandsData([selectedMarket.controllerAddress, BigInt(start), BigInt(pivot)]);
43
+ return pivotedBandsData;
44
+ }))).flat();
45
+
46
+ return bandsData.map((band) => ({
47
+ id: band.id.toString(),
48
+ collAmount: assetAmountInEth(band.collAmount.toString()),
49
+ debtAmount: assetAmountInEth(band.debtAmount.toString()),
50
+ lowPrice: assetAmountInEth(band.lowPrice.toString()),
51
+ highPrice: assetAmountInEth(band.highPrice.toString()),
52
+ }));
53
+ };
54
+
55
+ export const _getLlamaLendGlobalData = async (provider: Client, network: NetworkNumber, selectedMarket: LlamaLendMarketData): Promise<LlamaLendGlobalMarketData> => {
56
+ const contract = LlamaLendViewContractViem(provider, network);
57
+
58
+ const collAsset = selectedMarket.collAsset;
59
+ const debtAsset = selectedMarket.baseAsset;
60
+
61
+ const data = await contract.read.globalData([selectedMarket.controllerAddress]);
62
+
63
+ // all prices are in 18 decimals
64
+ const oraclePrice = getEthAmountForDecimals(data.oraclePrice.toString(), 18);
65
+ const collPriceUsd = collAsset === 'crvUSD' ? '1' : new Dec(1).mul(oraclePrice).toDP(18).toString();
66
+ const debtPriceUsd = debtAsset === 'crvUSD' ? '1' : new Dec(1).div(oraclePrice).toDP(18).toString();
67
+
68
+ const totalDebt = assetAmountInEth(data.totalDebt.toString(), debtAsset);
69
+ const totalDebtSupplied = assetAmountInEth(data.debtTokenTotalSupply.toString(), debtAsset);
70
+ const utilization = new Dec(totalDebtSupplied).gt(0)
71
+ ? new Dec(totalDebt).div(totalDebtSupplied).mul(100).toString()
72
+ : '0';
73
+ const ammPrice = assetAmountInEth(data.ammPrice.toString(), debtAsset);
74
+
75
+ const rate = assetAmountInEth(data.ammRate.toString());
76
+ const futureRate = assetAmountInEth(data.monetaryPolicyRate.toString());
77
+
78
+ const exponentRate = new Dec(rate).mul(365).mul(86400);
79
+ const exponentFutureRate = new Dec(futureRate).mul(365).mul(86400);
80
+ const borrowRate = new Dec(new Dec(2.718281828459).pow(exponentRate).minus(1)).mul(100)
81
+ .toString();
82
+ const futureBorrowRate = new Dec(new Dec(2.718281828459).pow(exponentFutureRate).minus(1)).mul(100)
83
+ .toString();
84
+
85
+ const bandsData = await getAndFormatBands(provider, network, selectedMarket, data.minBand.toString(), data.maxBand.toString());
86
+ const cap = assetAmountInEth(data.debtTokenTotalSupply.toString(), debtAsset);
87
+ const leftToBorrow = getEthAmountForDecimals(data.debtTokenLeftToBorrow.toString(), 18);
88
+
89
+ const debtInAYearBN = new Dec(totalDebt).mul(new Dec(2.718281828459).pow(exponentRate).toNumber());
90
+ const lendRate = debtInAYearBN.minus(totalDebt).div(cap).mul(100).toString();
91
+
92
+ const assetsData: LlamaLendAssetsData = {};
93
+ assetsData[debtAsset] = {
94
+ symbol: debtAsset,
95
+ address: data.debtToken,
96
+ price: debtPriceUsd,
97
+ supplyRate: lendRate,
98
+ borrowRate,
99
+ canBeSupplied: true,
100
+ canBeBorrowed: true,
101
+ supplyIncentives: [],
102
+ borrowIncentives: [],
103
+ };
104
+
105
+ assetsData[collAsset] = {
106
+ symbol: collAsset,
107
+ address: data.collateralToken,
108
+ price: collPriceUsd,
109
+ supplyRate: '0',
110
+ borrowRate: '0',
111
+ canBeSupplied: true,
112
+ canBeBorrowed: false,
113
+ supplyIncentives: [],
114
+ borrowIncentives: [],
115
+ };
116
+
117
+ if (STAKING_ASSETS.includes(collAsset)) {
118
+ assetsData[collAsset].supplyIncentives.push({
119
+ apy: await getStakingApy(collAsset),
120
+ token: collAsset,
121
+ incentiveKind: IncentiveKind.Staking,
122
+ description: `Native ${collAsset} yield.`,
123
+ });
124
+ }
125
+
126
+ return {
127
+ A: data.A.toString(),
128
+ loanDiscount: data.loanDiscount.toString(),
129
+ activeBand: data.activeBand.toString(),
130
+ monetaryPolicyRate: data.monetaryPolicyRate.toString(),
131
+ ammRate: data.ammRate.toString(),
132
+ minBand: data.minBand.toString(),
133
+ maxBand: data.maxBand.toString(),
134
+ assetsData,
135
+ totalDebt,
136
+ totalDebtSupplied,
137
+ utilization,
138
+ ammPrice,
139
+ oraclePrice: assetAmountInEth(data.oraclePrice.toString(), debtAsset),
140
+ basePrice: assetAmountInEth(data.basePrice.toString(), debtAsset),
141
+ minted: assetAmountInEth(data.minted.toString(), debtAsset),
142
+ redeemed: assetAmountInEth(data.redeemed.toString(), debtAsset),
143
+ borrowRate,
144
+ lendRate,
145
+ futureBorrowRate,
146
+ bands: bandsData,
147
+ leftToBorrow,
148
+ };
149
+ };
150
+
151
+ export const getLlamaLendGlobalData = async (
152
+ provider: EthereumProvider,
153
+ network: NetworkNumber,
154
+ selectedMarket: LlamaLendMarketData,
155
+ ): Promise<LlamaLendGlobalMarketData> => _getLlamaLendGlobalData(getViemProvider(provider, network), network, selectedMarket);
156
+
157
+ const getStatusForUser = (bandRange: string[], activeBand: string, debtSupplied: string, collSupplied: string, healthPercent: string) => {
158
+ // if bands are equal, that can only be [0,0] which means user doesn't have loan (min number of bands is 4)
159
+ if (new Dec(bandRange[0]).eq(bandRange[1])) return LlamaLendStatus.Nonexistant;
160
+ // if user doesn't have debtAsset as collateral, then his position is not in soft liquidation
161
+ if (new Dec(debtSupplied).lte(0)) {
162
+ const isHealthRisky = new Dec(healthPercent).lt(10);
163
+ if (new Dec(bandRange[0]).minus(activeBand).lte(3) || isHealthRisky) return LlamaLendStatus.Risk; // if user band is less than 3 bands away from active band, his position is at risk
164
+ return LlamaLendStatus.Safe;
165
+ }
166
+ if (new Dec(bandRange[0]).lte(activeBand) && new Dec(bandRange[1]).gte(activeBand)) return LlamaLendStatus.SoftLiquidating; // user has debtAsset as coll so he is in soft liquidation
167
+ if (new Dec(collSupplied).lte(0) || new Dec(bandRange[1]).lte(activeBand)) return LlamaLendStatus.SoftLiquidated; // or is fully soft liquidated
168
+ return LlamaLendStatus.Nonexistant;
169
+ };
170
+
171
+ export const _getLlamaLendAccountBalances = async (provider: Client, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress, controllerAddress: EthAddress): Promise<PositionBalances> => {
172
+ let balances: PositionBalances = {
173
+ collateral: {},
174
+ debt: {},
175
+ };
176
+
177
+ if (!address) {
178
+ return balances;
179
+ }
180
+
181
+ const contract = LlamaLendViewContractViem(provider, network, block);
182
+
183
+ const selectedMarket = getLlamaLendMarketFromControllerAddress(controllerAddress, network);
184
+ const data = await contract.read.userData([selectedMarket.controllerAddress, address], setViemBlockNumber(block));
185
+
186
+ balances = {
187
+ collateral: {
188
+ [addressMapping ? getAssetInfo(wethToEth(selectedMarket.collAsset), network).address.toLowerCase() : wethToEth(selectedMarket.collAsset)]: data.marketCollateralAmount.toString(),
189
+ },
190
+ debt: {
191
+ [addressMapping ? getAssetInfo(wethToEth(selectedMarket.baseAsset), network).address.toLowerCase() : wethToEth(selectedMarket.baseAsset)]: data.debtAmount.toString(),
192
+ },
193
+ };
194
+
195
+ return balances;
196
+ };
197
+
198
+ export const getLlamaLendAccountBalances = async (
199
+ provider: EthereumProvider,
200
+ network: NetworkNumber,
201
+ block: Blockish,
202
+ addressMapping: boolean,
203
+ address: EthAddress,
204
+ controllerAddress: EthAddress,
205
+ ): Promise<PositionBalances> => _getLlamaLendAccountBalances(getViemProvider(provider, network), network, block, addressMapping, address, controllerAddress);
206
+
207
+ export const _getLlamaLendUserData = async (provider: Client, network: NetworkNumber, address: EthAddress, selectedMarket: LlamaLendMarketData, marketData: LlamaLendGlobalMarketData): Promise<LlamaLendUserData> => {
208
+ const contract = LlamaLendViewContractViem(provider, network);
209
+ const { assetsData } = marketData;
210
+
211
+ const data = await contract.read.userData([selectedMarket.controllerAddress, address]);
212
+ const collAsset = selectedMarket.collAsset;
213
+ const debtAsset = selectedMarket.baseAsset;
214
+
215
+ const collPrice = assetsData[collAsset].price;
216
+ const debtPrice = assetsData[debtAsset].price;
217
+
218
+ const health = assetAmountInEth(data.health.toString());
219
+ const healthPercent = new Dec(health).mul(100).toString();
220
+
221
+ const collSupplied = assetAmountInEth(data.marketCollateralAmount.toString(), collAsset);
222
+ const collSuppliedUsd = new Dec(collSupplied).mul(collPrice).toString();
223
+
224
+ const debtSupplied = assetAmountInEth(data.debtTokenCollateralAmount.toString(), debtAsset);
225
+ const debtSuppliedUsd = new Dec(debtSupplied).mul(debtPrice).toString();
226
+
227
+ const debtSuppliedForYield = assetAmountInEth(data.debtTokenSuppliedAssets.toString(), debtAsset);
228
+ const debtSuppliedForYieldUsd = new Dec(debtSupplied).mul(debtPrice).toString();
229
+
230
+ const debtBorrowed = assetAmountInEth(data.debtAmount.toString(), debtAsset);
231
+ const debtBorrowedUsd = new Dec(debtBorrowed).mul(debtPrice).toString();
232
+ const shares = assetAmountInEth(data.debtTokenSuppliedShares.toString(), debtAsset);
233
+
234
+ const usedAssets: LlamaLendUsedAssets = {
235
+ [collAsset]: {
236
+ isSupplied: new Dec(collSupplied).gt('0'),
237
+ supplied: collSupplied,
238
+ suppliedUsd: collSuppliedUsd,
239
+ borrowed: '0',
240
+ borrowedUsd: '0',
241
+ isBorrowed: false,
242
+ symbol: collAsset,
243
+ collateral: true,
244
+ price: collPrice,
245
+ },
246
+ [debtAsset]: {
247
+ isSupplied: new Dec(debtSupplied).gt('0') || new Dec(debtSuppliedForYield).gt('0'),
248
+ collateral: new Dec(debtSupplied).gt('0'),
249
+ supplied: debtSupplied,
250
+ suppliedUsd: debtSuppliedUsd,
251
+ suppliedForYield: debtSuppliedForYield,
252
+ suppliedForYieldUsd: debtSuppliedForYieldUsd,
253
+ borrowed: debtBorrowed,
254
+ borrowedUsd: debtBorrowedUsd,
255
+ isBorrowed: new Dec(debtBorrowed).gt('0'),
256
+ symbol: debtAsset,
257
+ price: debtPrice,
258
+ shares,
259
+ },
260
+ };
261
+
262
+ const priceHigh = assetAmountInEth(data.priceHigh.toString());
263
+ const priceLow = assetAmountInEth(data.priceLow.toString());
264
+
265
+ const _userBands = data.loanExists ? (await getAndFormatBands(provider, network, selectedMarket, data.bandRange[0].toString(), data.bandRange[1].toString())) : [];
266
+
267
+ const status = data.loanExists ? getStatusForUser(data.bandRange.map(b => b.toString()), marketData.activeBand, debtSupplied, collSupplied, healthPercent) : LlamaLendStatus.Nonexistant;
268
+
269
+ const userBands = _userBands.map((band, index) => ({
270
+ ...band,
271
+ userDebtAmount: assetAmountInEth(data.usersBands[0][index].toString(), debtAsset),
272
+ userCollAmount: assetAmountInEth(data.usersBands[1][index].toString(), collAsset),
273
+ })).sort((a, b) => parseInt(b.id, 10) - parseInt(a.id, 10));
274
+
275
+ return {
276
+ ...data,
277
+ debtAmount: assetAmountInEth(data.debtAmount.toString(), debtAsset),
278
+ health,
279
+ healthPercent,
280
+ priceHigh,
281
+ priceLow,
282
+ liquidationDiscount: assetAmountInEth(data.liquidationDiscount.toString()),
283
+ numOfBands: data.N.toString(),
284
+ usedAssets,
285
+ status,
286
+ ...getLlamaLendAggregatedData({
287
+ loanExists: data.loanExists, usedAssets, network: NetworkNumber.Eth, selectedMarket, numOfBands: data.N.toString(), assetsData,
288
+ }),
289
+ userBands,
290
+ };
291
+ };
292
+
293
+ export const getLlamaLendUserData = async (
294
+ provider: EthereumProvider,
295
+ network: NetworkNumber,
296
+ address: EthAddress,
297
+ selectedMarket: LlamaLendMarketData,
298
+ marketData: LlamaLendGlobalMarketData,
299
+ ): Promise<LlamaLendUserData> => _getLlamaLendUserData(getViemProvider(provider, network), network, address, selectedMarket, marketData);
300
+
301
+ export const getLlamaLendFullPositionData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress, selectedMarket: LlamaLendMarketData): Promise<LlamaLendUserData> => {
302
+ const marketData = await getLlamaLendGlobalData(provider, network, selectedMarket);
303
+ const positionData = await getLlamaLendUserData(provider, network, address, selectedMarket, marketData);
304
+ return positionData;
305
+ };