@defisaver/positions-sdk 0.0.166-dev4-liquity-v2 → 0.0.166-dev4

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 (142) hide show
  1. package/.mocharc.json +4 -4
  2. package/.nvmrc +1 -1
  3. package/README.md +69 -69
  4. package/cjs/config/contracts.d.ts +32 -3
  5. package/cjs/config/contracts.js +3 -3
  6. package/cjs/contracts.d.ts +1 -1
  7. package/cjs/contracts.js +2 -2
  8. package/cjs/eulerV2/index.d.ts +41 -0
  9. package/cjs/eulerV2/index.js +218 -0
  10. package/cjs/helpers/eulerHelpers/index.d.ts +27 -0
  11. package/cjs/helpers/eulerHelpers/index.js +232 -0
  12. package/cjs/helpers/index.d.ts +1 -1
  13. package/cjs/helpers/index.js +2 -2
  14. package/cjs/index.d.ts +2 -2
  15. package/cjs/index.js +3 -3
  16. package/cjs/markets/euler/index.d.ts +8 -0
  17. package/cjs/markets/euler/index.js +30 -0
  18. package/cjs/markets/index.d.ts +1 -1
  19. package/cjs/markets/index.js +3 -3
  20. package/cjs/moneymarket/moneymarketCommonService.js +1 -1
  21. package/cjs/services/utils.d.ts +2 -0
  22. package/cjs/services/utils.js +4 -1
  23. package/cjs/types/contracts/generated/EulerV2View.d.ts +345 -0
  24. package/cjs/types/contracts/generated/index.d.ts +1 -1
  25. package/cjs/types/euler.d.ts +149 -0
  26. package/cjs/types/euler.js +14 -0
  27. package/cjs/types/index.d.ts +1 -1
  28. package/cjs/types/index.js +1 -1
  29. package/esm/config/contracts.d.ts +32 -3
  30. package/esm/config/contracts.js +3 -3
  31. package/esm/contracts.d.ts +1 -1
  32. package/esm/contracts.js +1 -1
  33. package/esm/eulerV2/index.d.ts +41 -0
  34. package/esm/eulerV2/index.js +210 -0
  35. package/esm/helpers/eulerHelpers/index.d.ts +27 -0
  36. package/esm/helpers/eulerHelpers/index.js +219 -0
  37. package/esm/helpers/index.d.ts +1 -1
  38. package/esm/helpers/index.js +1 -1
  39. package/esm/index.d.ts +2 -2
  40. package/esm/index.js +2 -2
  41. package/esm/markets/euler/index.d.ts +8 -0
  42. package/esm/markets/euler/index.js +24 -0
  43. package/esm/markets/index.d.ts +1 -1
  44. package/esm/markets/index.js +1 -1
  45. package/esm/moneymarket/moneymarketCommonService.js +1 -1
  46. package/esm/services/utils.d.ts +2 -0
  47. package/esm/services/utils.js +2 -0
  48. package/esm/types/contracts/generated/EulerV2View.d.ts +345 -0
  49. package/esm/types/contracts/generated/index.d.ts +1 -1
  50. package/esm/types/euler.d.ts +149 -0
  51. package/esm/types/euler.js +11 -0
  52. package/esm/types/index.d.ts +1 -1
  53. package/esm/types/index.js +1 -1
  54. package/package.json +49 -49
  55. package/src/aaveV2/index.ts +227 -227
  56. package/src/aaveV3/index.ts +590 -590
  57. package/src/assets/index.ts +60 -60
  58. package/src/chickenBonds/index.ts +123 -123
  59. package/src/compoundV2/index.ts +219 -219
  60. package/src/compoundV3/index.ts +281 -281
  61. package/src/config/contracts.js +1040 -1040
  62. package/src/constants/index.ts +6 -6
  63. package/src/contracts.ts +130 -130
  64. package/src/curveUsd/index.ts +229 -229
  65. package/src/eulerV2/index.ts +301 -0
  66. package/src/exchange/index.ts +17 -17
  67. package/src/helpers/aaveHelpers/index.ts +194 -194
  68. package/src/helpers/chickenBondsHelpers/index.ts +23 -23
  69. package/src/helpers/compoundHelpers/index.ts +246 -246
  70. package/src/helpers/curveUsdHelpers/index.ts +40 -40
  71. package/src/helpers/eulerHelpers/index.ts +231 -0
  72. package/src/helpers/index.ts +9 -9
  73. package/src/helpers/llamaLendHelpers/index.ts +53 -53
  74. package/src/helpers/makerHelpers/index.ts +94 -94
  75. package/src/helpers/morphoBlueHelpers/index.ts +115 -115
  76. package/src/helpers/sparkHelpers/index.ts +150 -150
  77. package/src/index.ts +48 -48
  78. package/src/liquity/index.ts +116 -116
  79. package/src/llamaLend/index.ts +275 -275
  80. package/src/maker/index.ts +117 -117
  81. package/src/markets/aave/index.ts +152 -152
  82. package/src/markets/aave/marketAssets.ts +46 -46
  83. package/src/markets/compound/index.ts +173 -173
  84. package/src/markets/compound/marketsAssets.ts +64 -64
  85. package/src/markets/curveUsd/index.ts +69 -69
  86. package/src/markets/euler/index.ts +27 -0
  87. package/src/markets/index.ts +24 -23
  88. package/src/markets/llamaLend/contractAddresses.ts +141 -141
  89. package/src/markets/llamaLend/index.ts +235 -235
  90. package/src/markets/morphoBlue/index.ts +728 -728
  91. package/src/markets/spark/index.ts +29 -29
  92. package/src/markets/spark/marketAssets.ts +10 -10
  93. package/src/moneymarket/moneymarketCommonService.ts +80 -80
  94. package/src/morphoAaveV2/index.ts +256 -256
  95. package/src/morphoAaveV3/index.ts +630 -630
  96. package/src/morphoBlue/index.ts +171 -171
  97. package/src/multicall/index.ts +22 -22
  98. package/src/services/dsrService.ts +15 -15
  99. package/src/services/priceService.ts +21 -21
  100. package/src/services/utils.ts +57 -54
  101. package/src/setup.ts +8 -8
  102. package/src/spark/index.ts +424 -424
  103. package/src/staking/staking.ts +218 -218
  104. package/src/types/aave.ts +262 -262
  105. package/src/types/chickenBonds.ts +45 -45
  106. package/src/types/common.ts +84 -84
  107. package/src/types/compound.ts +129 -129
  108. package/src/types/contracts/generated/EulerV2View.ts +446 -0
  109. package/src/types/contracts/generated/index.ts +1 -1
  110. package/src/types/curveUsd.ts +118 -118
  111. package/src/types/euler.ts +172 -0
  112. package/src/types/index.ts +10 -10
  113. package/src/types/liquity.ts +30 -30
  114. package/src/types/llamaLend.ts +155 -155
  115. package/src/types/maker.ts +50 -50
  116. package/src/types/morphoBlue.ts +146 -146
  117. package/src/types/spark.ts +127 -127
  118. package/cjs/helpers/liquityV2Helpers/index.d.ts +0 -12
  119. package/cjs/helpers/liquityV2Helpers/index.js +0 -62
  120. package/cjs/liquityV2/index.d.ts +0 -10
  121. package/cjs/liquityV2/index.js +0 -99
  122. package/cjs/markets/liquityV2/index.d.ts +0 -8
  123. package/cjs/markets/liquityV2/index.js +0 -34
  124. package/cjs/types/contracts/generated/LiquityV2View.d.ts +0 -222
  125. package/cjs/types/liquityV2.d.ts +0 -84
  126. package/cjs/types/liquityV2.js +0 -8
  127. package/esm/helpers/liquityV2Helpers/index.d.ts +0 -12
  128. package/esm/helpers/liquityV2Helpers/index.js +0 -54
  129. package/esm/liquityV2/index.d.ts +0 -10
  130. package/esm/liquityV2/index.js +0 -91
  131. package/esm/markets/liquityV2/index.d.ts +0 -8
  132. package/esm/markets/liquityV2/index.js +0 -28
  133. package/esm/types/contracts/generated/LiquityV2View.d.ts +0 -222
  134. package/esm/types/liquityV2.d.ts +0 -84
  135. package/esm/types/liquityV2.js +0 -5
  136. package/src/helpers/liquityV2Helpers/index.ts +0 -79
  137. package/src/liquityV2/index.ts +0 -116
  138. package/src/markets/liquityV2/index.ts +0 -31
  139. package/src/types/contracts/generated/LiquityV2View.ts +0 -280
  140. package/src/types/liquityV2.ts +0 -90
  141. /package/cjs/types/contracts/generated/{LiquityV2View.js → EulerV2View.js} +0 -0
  142. /package/esm/types/contracts/generated/{LiquityV2View.js → EulerV2View.js} +0 -0
@@ -1,275 +1,275 @@
1
- import Dec from 'decimal.js';
2
- import { assetAmountInEth, getAssetInfo } from '@defisaver/tokens';
3
- import Web3 from 'web3';
4
- import {
5
- BandData, LlamaLendGlobalMarketData, LlamaLendMarketData, LlamaLendStatus, LlamaLendUsedAssets, LlamaLendUserData,
6
- } from '../types';
7
- import { multicall } from '../multicall';
8
- import {
9
- Blockish, EthAddress, NetworkNumber, PositionBalances,
10
- } from '../types/common';
11
- import { getConfigContractAbi, getConfigContractAddress, LlamaLendViewContract } from '../contracts';
12
- import { getLlamaLendAggregatedData } from '../helpers/llamaLendHelpers';
13
- import { getAbiItem, getEthAmountForDecimals, wethToEth } from '../services/utils';
14
- import { USD_QUOTE } from '../constants';
15
- import { getLlamaLendMarketFromControllerAddress } from '../markets/llamaLend';
16
- import { getStakingApy, STAKING_ASSETS } from '../staking';
17
-
18
- const getAndFormatBands = async (web3: Web3, network: NetworkNumber, selectedMarket: LlamaLendMarketData, _minBand: string, _maxBand: string) => {
19
- const contract = LlamaLendViewContract(web3, 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
- // @ts-ignore
43
- const pivotedBandsData = await contract.methods.getBandsData(selectedMarket.controllerAddress, start, pivot).call();
44
- return pivotedBandsData;
45
- }))).flat();
46
-
47
- return bandsData.map((band: BandData) => ({
48
- id: band.id,
49
- collAmount: assetAmountInEth(band.collAmount),
50
- debtAmount: assetAmountInEth(band.debtAmount),
51
- lowPrice: assetAmountInEth(band.lowPrice),
52
- highPrice: assetAmountInEth(band.highPrice),
53
- }));
54
- };
55
-
56
- export const getLlamaLendGlobalData = async (web3: Web3, network: NetworkNumber, selectedMarket: LlamaLendMarketData, defaultWeb3: Web3): Promise<LlamaLendGlobalMarketData> => {
57
- const contract = LlamaLendViewContract(web3, network);
58
-
59
- const collAsset = selectedMarket.collAsset;
60
- const debtAsset = selectedMarket.baseAsset;
61
-
62
- const data = await contract.methods.globalData(selectedMarket.controllerAddress).call();
63
-
64
- // all prices are in 18 decimals
65
- const oraclePrice = getEthAmountForDecimals(data.oraclePrice, 18);
66
- const collPriceUsd = collAsset === 'crvUSD' ? '1' : new Dec(1).mul(oraclePrice).toDP(18).toString();
67
- const debtPriceUsd = debtAsset === 'crvUSD' ? '1' : new Dec(1).div(oraclePrice).toDP(18).toString();
68
-
69
- const totalDebt = assetAmountInEth(data.totalDebt, debtAsset);
70
- const totalDebtSupplied = assetAmountInEth(data.debtTokenTotalSupply, debtAsset);
71
- const utilization = new Dec(totalDebtSupplied).gt(0)
72
- ? new Dec(totalDebt).div(totalDebtSupplied).mul(100).toString()
73
- : '0';
74
- const ammPrice = assetAmountInEth(data.ammPrice, debtAsset);
75
-
76
- const rate = assetAmountInEth(data.ammRate);
77
- const futureRate = assetAmountInEth(data.monetaryPolicyRate);
78
-
79
- const exponentRate = new Dec(rate).mul(365).mul(86400);
80
- const exponentFutureRate = new Dec(futureRate).mul(365).mul(86400);
81
- const borrowRate = new Dec(new Dec(2.718281828459).pow(exponentRate).minus(1)).mul(100)
82
- .toString();
83
- const futureBorrowRate = new Dec(new Dec(2.718281828459).pow(exponentFutureRate).minus(1)).mul(100)
84
- .toString();
85
-
86
- const bandsData = await getAndFormatBands(web3, network, selectedMarket, data.minBand, data.maxBand);
87
- const cap = assetAmountInEth(data.debtTokenTotalSupply, debtAsset);
88
- const leftToBorrow = getEthAmountForDecimals(data.debtTokenLeftToBorrow, 18);
89
-
90
- const debtInAYearBN = new Dec(totalDebt).mul(new Dec(2.718281828459).pow(exponentRate).toNumber());
91
- const lendRate = debtInAYearBN.minus(totalDebt).div(cap).mul(100).toString();
92
-
93
- const assetsData:any = {};
94
- assetsData[debtAsset] = {
95
- symbol: debtAsset,
96
- address: data.debtToken,
97
- price: debtPriceUsd,
98
- supplyRate: lendRate,
99
- borrowRate,
100
- canBeSupplied: true,
101
- canBeBorrowed: true,
102
- };
103
-
104
- assetsData[collAsset] = {
105
- symbol: collAsset,
106
- address: data.collateralToken,
107
- price: collPriceUsd,
108
- supplyRate: '0',
109
- borrowRate: '0',
110
- canBeSupplied: true,
111
- canBeBorrowed: false,
112
- };
113
-
114
- if (STAKING_ASSETS.includes(collAsset)) {
115
- assetsData[collAsset].incentiveSupplyApy = await getStakingApy(collAsset, defaultWeb3);
116
- assetsData[collAsset].incentiveSupplyToken = collAsset;
117
- }
118
-
119
- return {
120
- A: data.A,
121
- loanDiscount: data.loanDiscount,
122
- activeBand: data.activeBand,
123
- monetaryPolicyRate: data.monetaryPolicyRate,
124
- ammRate: data.ammRate,
125
- minBand: data.minBand,
126
- maxBand: data.maxBand,
127
- assetsData,
128
- totalDebt,
129
- totalDebtSupplied,
130
- utilization,
131
- ammPrice,
132
- oraclePrice: assetAmountInEth(data.oraclePrice, debtAsset),
133
- basePrice: assetAmountInEth(data.basePrice, debtAsset),
134
- minted: assetAmountInEth(data.minted, debtAsset),
135
- redeemed: assetAmountInEth(data.redeemed, debtAsset),
136
- borrowRate,
137
- lendRate,
138
- futureBorrowRate,
139
- bands: bandsData,
140
- leftToBorrow,
141
- };
142
- };
143
-
144
- const getStatusForUser = (bandRange: string[], activeBand: string, debtSupplied: string, collSupplied: string, healthPercent: string) => {
145
- // if bands are equal, that can only be [0,0] which means user doesn't have loan (min number of bands is 4)
146
- if (new Dec(bandRange[0]).eq(bandRange[1])) return LlamaLendStatus.Nonexistant;
147
- // if user doesn't have debtAsset as collateral, then his position is not in soft liquidation
148
- if (new Dec(debtSupplied).lte(0)) {
149
- const isHealthRisky = new Dec(healthPercent).lt(10);
150
- 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
151
- return LlamaLendStatus.Safe;
152
- }
153
- 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
154
- if (new Dec(collSupplied).lte(0) || new Dec(bandRange[1]).lte(activeBand)) return LlamaLendStatus.SoftLiquidated; // or is fully soft liquidated
155
- return LlamaLendStatus.Nonexistant;
156
- };
157
-
158
- export const getLlamaLendAccountBalances = async (web3: Web3, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress, controllerAddress: EthAddress): Promise<PositionBalances> => {
159
- let balances: PositionBalances = {
160
- collateral: {},
161
- debt: {},
162
- };
163
-
164
- if (!address) {
165
- return balances;
166
- }
167
-
168
- const contract = LlamaLendViewContract(web3, network, block);
169
-
170
- const selectedMarket = getLlamaLendMarketFromControllerAddress(controllerAddress, network);
171
- const data = await contract.methods.userData(selectedMarket.controllerAddress, address).call({}, block);
172
-
173
- balances = {
174
- collateral: {
175
- [addressMapping ? getAssetInfo(wethToEth(selectedMarket.collAsset), network).address.toLowerCase() : wethToEth(selectedMarket.collAsset)]: data.marketCollateralAmount,
176
- },
177
- debt: {
178
- [addressMapping ? getAssetInfo(wethToEth(selectedMarket.baseAsset), network).address.toLowerCase() : wethToEth(selectedMarket.baseAsset)]: data.debtAmount,
179
- },
180
- };
181
-
182
- return balances;
183
- };
184
-
185
- export const getLlamaLendUserData = async (web3: Web3, network: NetworkNumber, address: string, selectedMarket: LlamaLendMarketData, marketData: LlamaLendGlobalMarketData): Promise<LlamaLendUserData> => {
186
- const contract = LlamaLendViewContract(web3, network);
187
- const { assetsData } = marketData;
188
-
189
- const data = await contract.methods.userData(selectedMarket.controllerAddress, address).call();
190
- const collAsset = selectedMarket.collAsset;
191
- const debtAsset = selectedMarket.baseAsset;
192
-
193
- const collPrice = assetsData[collAsset].price;
194
- const debtPrice = assetsData[debtAsset].price;
195
-
196
- const health = assetAmountInEth(data.health);
197
- const healthPercent = new Dec(health).mul(100).toString();
198
-
199
- const collSupplied = assetAmountInEth(data.marketCollateralAmount, collAsset);
200
- const collSuppliedUsd = new Dec(collSupplied).mul(collPrice).toString();
201
-
202
- const debtSupplied = assetAmountInEth(data.debtTokenCollateralAmount, debtAsset);
203
- const debtSuppliedUsd = new Dec(debtSupplied).mul(debtPrice).toString();
204
-
205
- const debtSuppliedForYield = assetAmountInEth(data.debtTokenSuppliedAssets, debtAsset);
206
- const debtSuppliedForYieldUsd = new Dec(debtSupplied).mul(debtPrice).toString();
207
-
208
- const debtBorrowed = assetAmountInEth(data.debtAmount, debtAsset);
209
- const debtBorrowedUsd = new Dec(debtBorrowed).mul(debtPrice).toString();
210
- const shares = assetAmountInEth(data.debtTokenSuppliedShares, debtAsset);
211
-
212
- const usedAssets: LlamaLendUsedAssets = {
213
- [collAsset]: {
214
- isSupplied: new Dec(collSupplied).gt('0'),
215
- supplied: collSupplied,
216
- suppliedUsd: collSuppliedUsd,
217
- borrowed: '0',
218
- borrowedUsd: '0',
219
- isBorrowed: false,
220
- symbol: collAsset,
221
- collateral: true,
222
- price: collPrice,
223
- },
224
- [debtAsset]: {
225
- isSupplied: new Dec(debtSupplied).gt('0') || new Dec(debtSuppliedForYield).gt('0'),
226
- collateral: new Dec(debtSupplied).gt('0'),
227
- supplied: debtSupplied,
228
- suppliedUsd: debtSuppliedUsd,
229
- suppliedForYield: debtSuppliedForYield,
230
- suppliedForYieldUsd: debtSuppliedForYieldUsd,
231
- borrowed: debtBorrowed,
232
- borrowedUsd: debtBorrowedUsd,
233
- isBorrowed: new Dec(debtBorrowed).gt('0'),
234
- symbol: debtAsset,
235
- price: debtPrice,
236
- shares,
237
- },
238
- };
239
-
240
- const priceHigh = assetAmountInEth(data.priceHigh);
241
- const priceLow = assetAmountInEth(data.priceLow);
242
-
243
- const _userBands = data.loanExists ? (await getAndFormatBands(web3, network, selectedMarket, data.bandRange[0], data.bandRange[1])) : [];
244
-
245
- const status = data.loanExists ? getStatusForUser(data.bandRange, marketData.activeBand, debtSupplied, collSupplied, healthPercent) : LlamaLendStatus.Nonexistant;
246
-
247
- const userBands = _userBands.map((band, index) => ({
248
- ...band,
249
- userDebtAmount: assetAmountInEth(data.usersBands[0][index], debtAsset),
250
- userCollAmount: assetAmountInEth(data.usersBands[1][index], collAsset),
251
- })).sort((a, b) => parseInt(b.id, 10) - parseInt(a.id, 10));
252
-
253
- return {
254
- ...data,
255
- debtAmount: assetAmountInEth(data.debtAmount, debtAsset),
256
- health,
257
- healthPercent,
258
- priceHigh,
259
- priceLow,
260
- liquidationDiscount: assetAmountInEth(data.liquidationDiscount),
261
- numOfBands: data.N,
262
- usedAssets,
263
- status,
264
- ...getLlamaLendAggregatedData({
265
- loanExists: data.loanExists, usedAssets, network: NetworkNumber.Eth, selectedMarket, numOfBands: data.N, assetsData,
266
- }),
267
- userBands,
268
- };
269
- };
270
-
271
- export const getLlamaLendFullPositionData = async (web3: Web3, network: NetworkNumber, address: string, selectedMarket: LlamaLendMarketData, defaultWeb3: Web3): Promise<LlamaLendUserData> => {
272
- const marketData = await getLlamaLendGlobalData(web3, network, selectedMarket, defaultWeb3);
273
- const positionData = await getLlamaLendUserData(web3, network, address, selectedMarket, marketData);
274
- return positionData;
275
- };
1
+ import Dec from 'decimal.js';
2
+ import { assetAmountInEth, getAssetInfo } from '@defisaver/tokens';
3
+ import Web3 from 'web3';
4
+ import {
5
+ BandData, LlamaLendGlobalMarketData, LlamaLendMarketData, LlamaLendStatus, LlamaLendUsedAssets, LlamaLendUserData,
6
+ } from '../types';
7
+ import { multicall } from '../multicall';
8
+ import {
9
+ Blockish, EthAddress, NetworkNumber, PositionBalances,
10
+ } from '../types/common';
11
+ import { getConfigContractAbi, getConfigContractAddress, LlamaLendViewContract } from '../contracts';
12
+ import { getLlamaLendAggregatedData } from '../helpers/llamaLendHelpers';
13
+ import { getAbiItem, getEthAmountForDecimals, wethToEth } from '../services/utils';
14
+ import { USD_QUOTE } from '../constants';
15
+ import { getLlamaLendMarketFromControllerAddress } from '../markets/llamaLend';
16
+ import { getStakingApy, STAKING_ASSETS } from '../staking';
17
+
18
+ const getAndFormatBands = async (web3: Web3, network: NetworkNumber, selectedMarket: LlamaLendMarketData, _minBand: string, _maxBand: string) => {
19
+ const contract = LlamaLendViewContract(web3, 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
+ // @ts-ignore
43
+ const pivotedBandsData = await contract.methods.getBandsData(selectedMarket.controllerAddress, start, pivot).call();
44
+ return pivotedBandsData;
45
+ }))).flat();
46
+
47
+ return bandsData.map((band: BandData) => ({
48
+ id: band.id,
49
+ collAmount: assetAmountInEth(band.collAmount),
50
+ debtAmount: assetAmountInEth(band.debtAmount),
51
+ lowPrice: assetAmountInEth(band.lowPrice),
52
+ highPrice: assetAmountInEth(band.highPrice),
53
+ }));
54
+ };
55
+
56
+ export const getLlamaLendGlobalData = async (web3: Web3, network: NetworkNumber, selectedMarket: LlamaLendMarketData, defaultWeb3: Web3): Promise<LlamaLendGlobalMarketData> => {
57
+ const contract = LlamaLendViewContract(web3, network);
58
+
59
+ const collAsset = selectedMarket.collAsset;
60
+ const debtAsset = selectedMarket.baseAsset;
61
+
62
+ const data = await contract.methods.globalData(selectedMarket.controllerAddress).call();
63
+
64
+ // all prices are in 18 decimals
65
+ const oraclePrice = getEthAmountForDecimals(data.oraclePrice, 18);
66
+ const collPriceUsd = collAsset === 'crvUSD' ? '1' : new Dec(1).mul(oraclePrice).toDP(18).toString();
67
+ const debtPriceUsd = debtAsset === 'crvUSD' ? '1' : new Dec(1).div(oraclePrice).toDP(18).toString();
68
+
69
+ const totalDebt = assetAmountInEth(data.totalDebt, debtAsset);
70
+ const totalDebtSupplied = assetAmountInEth(data.debtTokenTotalSupply, debtAsset);
71
+ const utilization = new Dec(totalDebtSupplied).gt(0)
72
+ ? new Dec(totalDebt).div(totalDebtSupplied).mul(100).toString()
73
+ : '0';
74
+ const ammPrice = assetAmountInEth(data.ammPrice, debtAsset);
75
+
76
+ const rate = assetAmountInEth(data.ammRate);
77
+ const futureRate = assetAmountInEth(data.monetaryPolicyRate);
78
+
79
+ const exponentRate = new Dec(rate).mul(365).mul(86400);
80
+ const exponentFutureRate = new Dec(futureRate).mul(365).mul(86400);
81
+ const borrowRate = new Dec(new Dec(2.718281828459).pow(exponentRate).minus(1)).mul(100)
82
+ .toString();
83
+ const futureBorrowRate = new Dec(new Dec(2.718281828459).pow(exponentFutureRate).minus(1)).mul(100)
84
+ .toString();
85
+
86
+ const bandsData = await getAndFormatBands(web3, network, selectedMarket, data.minBand, data.maxBand);
87
+ const cap = assetAmountInEth(data.debtTokenTotalSupply, debtAsset);
88
+ const leftToBorrow = getEthAmountForDecimals(data.debtTokenLeftToBorrow, 18);
89
+
90
+ const debtInAYearBN = new Dec(totalDebt).mul(new Dec(2.718281828459).pow(exponentRate).toNumber());
91
+ const lendRate = debtInAYearBN.minus(totalDebt).div(cap).mul(100).toString();
92
+
93
+ const assetsData:any = {};
94
+ assetsData[debtAsset] = {
95
+ symbol: debtAsset,
96
+ address: data.debtToken,
97
+ price: debtPriceUsd,
98
+ supplyRate: lendRate,
99
+ borrowRate,
100
+ canBeSupplied: true,
101
+ canBeBorrowed: true,
102
+ };
103
+
104
+ assetsData[collAsset] = {
105
+ symbol: collAsset,
106
+ address: data.collateralToken,
107
+ price: collPriceUsd,
108
+ supplyRate: '0',
109
+ borrowRate: '0',
110
+ canBeSupplied: true,
111
+ canBeBorrowed: false,
112
+ };
113
+
114
+ if (STAKING_ASSETS.includes(collAsset)) {
115
+ assetsData[collAsset].incentiveSupplyApy = await getStakingApy(collAsset, defaultWeb3);
116
+ assetsData[collAsset].incentiveSupplyToken = collAsset;
117
+ }
118
+
119
+ return {
120
+ A: data.A,
121
+ loanDiscount: data.loanDiscount,
122
+ activeBand: data.activeBand,
123
+ monetaryPolicyRate: data.monetaryPolicyRate,
124
+ ammRate: data.ammRate,
125
+ minBand: data.minBand,
126
+ maxBand: data.maxBand,
127
+ assetsData,
128
+ totalDebt,
129
+ totalDebtSupplied,
130
+ utilization,
131
+ ammPrice,
132
+ oraclePrice: assetAmountInEth(data.oraclePrice, debtAsset),
133
+ basePrice: assetAmountInEth(data.basePrice, debtAsset),
134
+ minted: assetAmountInEth(data.minted, debtAsset),
135
+ redeemed: assetAmountInEth(data.redeemed, debtAsset),
136
+ borrowRate,
137
+ lendRate,
138
+ futureBorrowRate,
139
+ bands: bandsData,
140
+ leftToBorrow,
141
+ };
142
+ };
143
+
144
+ const getStatusForUser = (bandRange: string[], activeBand: string, debtSupplied: string, collSupplied: string, healthPercent: string) => {
145
+ // if bands are equal, that can only be [0,0] which means user doesn't have loan (min number of bands is 4)
146
+ if (new Dec(bandRange[0]).eq(bandRange[1])) return LlamaLendStatus.Nonexistant;
147
+ // if user doesn't have debtAsset as collateral, then his position is not in soft liquidation
148
+ if (new Dec(debtSupplied).lte(0)) {
149
+ const isHealthRisky = new Dec(healthPercent).lt(10);
150
+ 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
151
+ return LlamaLendStatus.Safe;
152
+ }
153
+ 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
154
+ if (new Dec(collSupplied).lte(0) || new Dec(bandRange[1]).lte(activeBand)) return LlamaLendStatus.SoftLiquidated; // or is fully soft liquidated
155
+ return LlamaLendStatus.Nonexistant;
156
+ };
157
+
158
+ export const getLlamaLendAccountBalances = async (web3: Web3, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress, controllerAddress: EthAddress): Promise<PositionBalances> => {
159
+ let balances: PositionBalances = {
160
+ collateral: {},
161
+ debt: {},
162
+ };
163
+
164
+ if (!address) {
165
+ return balances;
166
+ }
167
+
168
+ const contract = LlamaLendViewContract(web3, network, block);
169
+
170
+ const selectedMarket = getLlamaLendMarketFromControllerAddress(controllerAddress, network);
171
+ const data = await contract.methods.userData(selectedMarket.controllerAddress, address).call({}, block);
172
+
173
+ balances = {
174
+ collateral: {
175
+ [addressMapping ? getAssetInfo(wethToEth(selectedMarket.collAsset), network).address.toLowerCase() : wethToEth(selectedMarket.collAsset)]: data.marketCollateralAmount,
176
+ },
177
+ debt: {
178
+ [addressMapping ? getAssetInfo(wethToEth(selectedMarket.baseAsset), network).address.toLowerCase() : wethToEth(selectedMarket.baseAsset)]: data.debtAmount,
179
+ },
180
+ };
181
+
182
+ return balances;
183
+ };
184
+
185
+ export const getLlamaLendUserData = async (web3: Web3, network: NetworkNumber, address: string, selectedMarket: LlamaLendMarketData, marketData: LlamaLendGlobalMarketData): Promise<LlamaLendUserData> => {
186
+ const contract = LlamaLendViewContract(web3, network);
187
+ const { assetsData } = marketData;
188
+
189
+ const data = await contract.methods.userData(selectedMarket.controllerAddress, address).call();
190
+ const collAsset = selectedMarket.collAsset;
191
+ const debtAsset = selectedMarket.baseAsset;
192
+
193
+ const collPrice = assetsData[collAsset].price;
194
+ const debtPrice = assetsData[debtAsset].price;
195
+
196
+ const health = assetAmountInEth(data.health);
197
+ const healthPercent = new Dec(health).mul(100).toString();
198
+
199
+ const collSupplied = assetAmountInEth(data.marketCollateralAmount, collAsset);
200
+ const collSuppliedUsd = new Dec(collSupplied).mul(collPrice).toString();
201
+
202
+ const debtSupplied = assetAmountInEth(data.debtTokenCollateralAmount, debtAsset);
203
+ const debtSuppliedUsd = new Dec(debtSupplied).mul(debtPrice).toString();
204
+
205
+ const debtSuppliedForYield = assetAmountInEth(data.debtTokenSuppliedAssets, debtAsset);
206
+ const debtSuppliedForYieldUsd = new Dec(debtSupplied).mul(debtPrice).toString();
207
+
208
+ const debtBorrowed = assetAmountInEth(data.debtAmount, debtAsset);
209
+ const debtBorrowedUsd = new Dec(debtBorrowed).mul(debtPrice).toString();
210
+ const shares = assetAmountInEth(data.debtTokenSuppliedShares, debtAsset);
211
+
212
+ const usedAssets: LlamaLendUsedAssets = {
213
+ [collAsset]: {
214
+ isSupplied: new Dec(collSupplied).gt('0'),
215
+ supplied: collSupplied,
216
+ suppliedUsd: collSuppliedUsd,
217
+ borrowed: '0',
218
+ borrowedUsd: '0',
219
+ isBorrowed: false,
220
+ symbol: collAsset,
221
+ collateral: true,
222
+ price: collPrice,
223
+ },
224
+ [debtAsset]: {
225
+ isSupplied: new Dec(debtSupplied).gt('0') || new Dec(debtSuppliedForYield).gt('0'),
226
+ collateral: new Dec(debtSupplied).gt('0'),
227
+ supplied: debtSupplied,
228
+ suppliedUsd: debtSuppliedUsd,
229
+ suppliedForYield: debtSuppliedForYield,
230
+ suppliedForYieldUsd: debtSuppliedForYieldUsd,
231
+ borrowed: debtBorrowed,
232
+ borrowedUsd: debtBorrowedUsd,
233
+ isBorrowed: new Dec(debtBorrowed).gt('0'),
234
+ symbol: debtAsset,
235
+ price: debtPrice,
236
+ shares,
237
+ },
238
+ };
239
+
240
+ const priceHigh = assetAmountInEth(data.priceHigh);
241
+ const priceLow = assetAmountInEth(data.priceLow);
242
+
243
+ const _userBands = data.loanExists ? (await getAndFormatBands(web3, network, selectedMarket, data.bandRange[0], data.bandRange[1])) : [];
244
+
245
+ const status = data.loanExists ? getStatusForUser(data.bandRange, marketData.activeBand, debtSupplied, collSupplied, healthPercent) : LlamaLendStatus.Nonexistant;
246
+
247
+ const userBands = _userBands.map((band, index) => ({
248
+ ...band,
249
+ userDebtAmount: assetAmountInEth(data.usersBands[0][index], debtAsset),
250
+ userCollAmount: assetAmountInEth(data.usersBands[1][index], collAsset),
251
+ })).sort((a, b) => parseInt(b.id, 10) - parseInt(a.id, 10));
252
+
253
+ return {
254
+ ...data,
255
+ debtAmount: assetAmountInEth(data.debtAmount, debtAsset),
256
+ health,
257
+ healthPercent,
258
+ priceHigh,
259
+ priceLow,
260
+ liquidationDiscount: assetAmountInEth(data.liquidationDiscount),
261
+ numOfBands: data.N,
262
+ usedAssets,
263
+ status,
264
+ ...getLlamaLendAggregatedData({
265
+ loanExists: data.loanExists, usedAssets, network: NetworkNumber.Eth, selectedMarket, numOfBands: data.N, assetsData,
266
+ }),
267
+ userBands,
268
+ };
269
+ };
270
+
271
+ export const getLlamaLendFullPositionData = async (web3: Web3, network: NetworkNumber, address: string, selectedMarket: LlamaLendMarketData, defaultWeb3: Web3): Promise<LlamaLendUserData> => {
272
+ const marketData = await getLlamaLendGlobalData(web3, network, selectedMarket, defaultWeb3);
273
+ const positionData = await getLlamaLendUserData(web3, network, address, selectedMarket, marketData);
274
+ return positionData;
275
+ };