@defisaver/positions-sdk 0.0.198 → 0.0.199-dev-2

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 (133) 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 +126 -4
  5. package/cjs/config/contracts.js +14 -0
  6. package/cjs/contracts.d.ts +1 -0
  7. package/cjs/contracts.js +2 -1
  8. package/cjs/helpers/index.d.ts +1 -0
  9. package/cjs/helpers/index.js +2 -1
  10. package/cjs/helpers/liquityV2Helpers/index.d.ts +12 -0
  11. package/cjs/helpers/liquityV2Helpers/index.js +63 -0
  12. package/cjs/helpers/morphoBlueHelpers/index.js +66 -66
  13. package/cjs/index.d.ts +2 -1
  14. package/cjs/index.js +3 -1
  15. package/cjs/liquityV2/index.d.ts +18 -0
  16. package/cjs/liquityV2/index.js +184 -0
  17. package/cjs/markets/aave/marketAssets.js +2 -4
  18. package/cjs/markets/index.d.ts +1 -0
  19. package/cjs/markets/index.js +3 -1
  20. package/cjs/markets/liquityV2/index.d.ts +10 -0
  21. package/cjs/markets/liquityV2/index.js +47 -0
  22. package/cjs/moneymarket/moneymarketCommonService.js +1 -1
  23. package/cjs/types/contracts/generated/LiquityV2CollSurplusPool.d.ts +64 -0
  24. package/cjs/types/contracts/generated/LiquityV2CollSurplusPool.js +5 -0
  25. package/cjs/types/contracts/generated/LiquityV2TroveNFT.d.ts +73 -0
  26. package/cjs/types/contracts/generated/LiquityV2TroveNFT.js +5 -0
  27. package/cjs/types/contracts/generated/LiquityV2View.d.ts +244 -0
  28. package/cjs/types/contracts/generated/LiquityV2View.js +5 -0
  29. package/cjs/types/contracts/generated/index.d.ts +3 -0
  30. package/cjs/types/index.d.ts +1 -0
  31. package/cjs/types/index.js +1 -0
  32. package/cjs/types/liquityV2.d.ts +111 -0
  33. package/cjs/types/liquityV2.js +24 -0
  34. package/esm/config/contracts.d.ts +126 -4
  35. package/esm/config/contracts.js +14 -0
  36. package/esm/contracts.d.ts +1 -0
  37. package/esm/contracts.js +1 -0
  38. package/esm/helpers/index.d.ts +1 -0
  39. package/esm/helpers/index.js +1 -0
  40. package/esm/helpers/liquityV2Helpers/index.d.ts +12 -0
  41. package/esm/helpers/liquityV2Helpers/index.js +55 -0
  42. package/esm/helpers/morphoBlueHelpers/index.js +66 -66
  43. package/esm/index.d.ts +2 -1
  44. package/esm/index.js +2 -1
  45. package/esm/liquityV2/index.d.ts +18 -0
  46. package/esm/liquityV2/index.js +173 -0
  47. package/esm/markets/aave/marketAssets.js +2 -4
  48. package/esm/markets/index.d.ts +1 -0
  49. package/esm/markets/index.js +1 -0
  50. package/esm/markets/liquityV2/index.d.ts +10 -0
  51. package/esm/markets/liquityV2/index.js +40 -0
  52. package/esm/moneymarket/moneymarketCommonService.js +1 -1
  53. package/esm/types/contracts/generated/LiquityV2CollSurplusPool.d.ts +64 -0
  54. package/esm/types/contracts/generated/LiquityV2CollSurplusPool.js +4 -0
  55. package/esm/types/contracts/generated/LiquityV2TroveNFT.d.ts +73 -0
  56. package/esm/types/contracts/generated/LiquityV2TroveNFT.js +4 -0
  57. package/esm/types/contracts/generated/LiquityV2View.d.ts +244 -0
  58. package/esm/types/contracts/generated/LiquityV2View.js +4 -0
  59. package/esm/types/contracts/generated/index.d.ts +3 -0
  60. package/esm/types/index.d.ts +1 -0
  61. package/esm/types/index.js +1 -0
  62. package/esm/types/liquityV2.d.ts +111 -0
  63. package/esm/types/liquityV2.js +21 -0
  64. package/package.json +49 -49
  65. package/src/aaveV2/index.ts +227 -227
  66. package/src/aaveV3/index.ts +625 -625
  67. package/src/assets/index.ts +60 -60
  68. package/src/chickenBonds/index.ts +123 -123
  69. package/src/compoundV2/index.ts +220 -220
  70. package/src/compoundV3/index.ts +291 -291
  71. package/src/config/contracts.js +1109 -1095
  72. package/src/constants/index.ts +6 -6
  73. package/src/contracts.ts +133 -132
  74. package/src/curveUsd/index.ts +229 -229
  75. package/src/eulerV2/index.ts +303 -303
  76. package/src/exchange/index.ts +17 -17
  77. package/src/helpers/aaveHelpers/index.ts +198 -198
  78. package/src/helpers/chickenBondsHelpers/index.ts +23 -23
  79. package/src/helpers/compoundHelpers/index.ts +246 -246
  80. package/src/helpers/curveUsdHelpers/index.ts +40 -40
  81. package/src/helpers/eulerHelpers/index.ts +232 -232
  82. package/src/helpers/index.ts +10 -9
  83. package/src/helpers/liquityV2Helpers/index.ts +80 -0
  84. package/src/helpers/llamaLendHelpers/index.ts +53 -53
  85. package/src/helpers/makerHelpers/index.ts +94 -94
  86. package/src/helpers/morphoBlueHelpers/index.ts +365 -365
  87. package/src/helpers/sparkHelpers/index.ts +150 -150
  88. package/src/index.ts +50 -48
  89. package/src/liquity/index.ts +116 -116
  90. package/src/liquityV2/index.ts +220 -0
  91. package/src/llamaLend/index.ts +275 -275
  92. package/src/maker/index.ts +117 -117
  93. package/src/markets/aave/index.ts +152 -152
  94. package/src/markets/aave/marketAssets.ts +44 -46
  95. package/src/markets/compound/index.ts +213 -213
  96. package/src/markets/compound/marketsAssets.ts +82 -82
  97. package/src/markets/curveUsd/index.ts +69 -69
  98. package/src/markets/euler/index.ts +26 -26
  99. package/src/markets/index.ts +24 -24
  100. package/src/markets/liquityV2/index.ts +44 -0
  101. package/src/markets/llamaLend/contractAddresses.ts +141 -141
  102. package/src/markets/llamaLend/index.ts +235 -235
  103. package/src/markets/morphoBlue/index.ts +895 -895
  104. package/src/markets/spark/index.ts +29 -29
  105. package/src/markets/spark/marketAssets.ts +10 -10
  106. package/src/moneymarket/moneymarketCommonService.ts +80 -80
  107. package/src/morphoAaveV2/index.ts +256 -256
  108. package/src/morphoAaveV3/index.ts +631 -631
  109. package/src/morphoBlue/index.ts +204 -204
  110. package/src/multicall/index.ts +22 -22
  111. package/src/services/dsrService.ts +15 -15
  112. package/src/services/priceService.ts +62 -62
  113. package/src/services/utils.ts +56 -56
  114. package/src/setup.ts +8 -8
  115. package/src/spark/index.ts +461 -461
  116. package/src/staking/staking.ts +220 -220
  117. package/src/types/aave.ts +271 -271
  118. package/src/types/chickenBonds.ts +45 -45
  119. package/src/types/common.ts +84 -84
  120. package/src/types/compound.ts +131 -131
  121. package/src/types/contracts/generated/LiquityV2CollSurplusPool.ts +130 -0
  122. package/src/types/contracts/generated/LiquityV2TroveNFT.ts +150 -0
  123. package/src/types/contracts/generated/LiquityV2View.ts +315 -0
  124. package/src/types/contracts/generated/index.ts +3 -0
  125. package/src/types/curveUsd.ts +118 -118
  126. package/src/types/euler.ts +171 -171
  127. package/src/types/index.ts +10 -9
  128. package/src/types/liquity.ts +30 -30
  129. package/src/types/liquityV2.ts +119 -0
  130. package/src/types/llamaLend.ts +155 -155
  131. package/src/types/maker.ts +50 -50
  132. package/src/types/morphoBlue.ts +192 -192
  133. package/src/types/spark.ts +131 -131
@@ -1,230 +1,230 @@
1
- import Dec from 'decimal.js';
2
- import { assetAmountInEth, getAssetInfo } from '@defisaver/tokens';
3
- import Web3 from 'web3';
4
- import {
5
- BandData, CrvUSDGlobalMarketData, CrvUSDMarketData, CrvUSDStatus, CrvUSDUsedAssets, CrvUSDUserData, CrvUSDVersions,
6
- } from '../types';
7
- import { multicall } from '../multicall';
8
- import {
9
- Blockish, EthAddress, NetworkNumber, PositionBalances,
10
- } from '../types/common';
11
- import { CrvUSDFactoryContract, CrvUSDViewContract } from '../contracts';
12
- import { getCrvUsdAggregatedData } from '../helpers/curveUsdHelpers';
13
- import { CrvUsdMarkets } from '../markets';
14
- import { wethToEth } from '../services/utils';
15
-
16
- const getAndFormatBands = async (web3: Web3, network: NetworkNumber, selectedMarket: CrvUSDMarketData, _minBand: string, _maxBand: string) => {
17
- const contract = CrvUSDViewContract(web3, network);
18
- const minBand = parseInt(_minBand, 10);
19
- const maxBand = parseInt(_maxBand, 10);
20
- const pivots: number[] = [];
21
-
22
- // 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
23
- let i = minBand;
24
- while (i < maxBand) {
25
- i += 200;
26
- if (i > maxBand) {
27
- pivots.push(maxBand);
28
- } else {
29
- pivots.push(i);
30
- }
31
- }
32
-
33
- const bandsData = (await Promise.all(pivots.map(async (pivot, index) => {
34
- let start = 0;
35
- if (index === 0) {
36
- start = minBand;
37
- } else {
38
- start = pivots[index - 1] + 1;
39
- }
40
- // @ts-ignore
41
- const pivotedBandsData = await contract.methods.getBandsData(selectedMarket.controllerAddress, start, pivot).call();
42
- return pivotedBandsData;
43
- }))).flat();
44
-
45
- return bandsData.map((band: BandData) => ({
46
- id: band.id,
47
- collAmount: assetAmountInEth(band.collAmount),
48
- debtAmount: assetAmountInEth(band.debtAmount),
49
- lowPrice: assetAmountInEth(band.lowPrice),
50
- highPrice: assetAmountInEth(band.highPrice),
51
- }));
52
- };
53
-
54
- export const getCurveUsdGlobalData = async (web3: Web3, network: NetworkNumber, selectedMarket: CrvUSDMarketData): Promise<CrvUSDGlobalMarketData> => {
55
- const contract = CrvUSDViewContract(web3, network);
56
- const factoryContract = CrvUSDFactoryContract(web3, network);
57
- const collAsset = selectedMarket.collAsset;
58
- const debtAsset = selectedMarket.baseAsset;
59
-
60
- const multicallData = [
61
- {
62
- target: factoryContract.options.address,
63
- abiItem: factoryContract.options.jsonInterface.find(({ name }) => name === 'debt_ceiling'),
64
- params: [selectedMarket.controllerAddress],
65
- },
66
- {
67
- target: factoryContract.options.address,
68
- abiItem: factoryContract.options.jsonInterface.find(({ name }) => name === 'total_debt'),
69
- params: [],
70
- },
71
- {
72
- target: contract.options.address,
73
- abiItem: contract.options.jsonInterface.find(({ name }) => name === 'globalData'),
74
- params: [selectedMarket.controllerAddress],
75
- },
76
- ];
77
- const multiRes = await multicall(multicallData, web3, network);
78
- const data = multiRes[2][0];
79
- const debtCeiling = assetAmountInEth(multiRes[0][0], debtAsset);
80
-
81
- // all prices are in 18 decimals
82
- const totalDebt = assetAmountInEth(data.totalDebt, debtAsset);
83
- const ammPrice = assetAmountInEth(data.ammPrice, debtAsset);
84
-
85
- const rate = assetAmountInEth(data.ammRate);
86
- const futureRate = assetAmountInEth(data.monetaryPolicyRate);
87
-
88
- const exponentRate = new Dec(rate).mul(365).mul(86400);
89
- const exponentFutureRate = new Dec(futureRate).mul(365).mul(86400);
90
- const borrowRate = new Dec(new Dec(2.718281828459).pow(exponentRate).minus(1)).mul(100)
91
- .toString();
92
- const futureBorrowRate = new Dec(new Dec(2.718281828459).pow(exponentFutureRate).minus(1)).mul(100)
93
- .toString();
94
-
95
- const bandsData = await getAndFormatBands(web3, network, selectedMarket, data.minBand, data.maxBand);
96
-
97
- const leftToBorrow = new Dec(debtCeiling).minus(totalDebt).toString();
98
- return {
99
- ...data,
100
- debtCeiling,
101
- totalDebt,
102
- ammPrice,
103
- oraclePrice: assetAmountInEth(data.oraclePrice, debtAsset),
104
- basePrice: assetAmountInEth(data.basePrice, debtAsset),
105
- minted: assetAmountInEth(data.minted, debtAsset),
106
- redeemed: assetAmountInEth(data.redeemed, debtAsset),
107
- borrowRate,
108
- futureBorrowRate,
109
- bands: bandsData,
110
- leftToBorrow,
111
- };
112
- };
113
-
114
- const getStatusForUser = (bandRange: string[], activeBand: string, crvUSDSupplied: string, collSupplied: string, healthPercent: string) => {
115
- // if bands are equal, that can only be [0,0] which means user doesn't have loan (min number of bands is 4)
116
- if (new Dec(bandRange[0]).eq(bandRange[1])) return CrvUSDStatus.Nonexistant;
117
- // if user doesn't have crvUSD as collateral, then his position is not in soft liquidation
118
- if (new Dec(crvUSDSupplied).lte(0)) {
119
- const isHealthRisky = new Dec(healthPercent).lt(10);
120
- if (new Dec(bandRange[0]).minus(activeBand).lte(3) || isHealthRisky) return CrvUSDStatus.Risk; // if user band is less than 3 bands away from active band, his position is at risk
121
- return CrvUSDStatus.Safe;
122
- }
123
- if (new Dec(bandRange[0]).lte(activeBand) && new Dec(bandRange[1]).gte(activeBand)) return CrvUSDStatus.SoftLiquidating; // user has crvUSD as coll so he is in soft liquidation
124
- if (new Dec(collSupplied).lte(0) || new Dec(bandRange[1]).lte(activeBand)) return CrvUSDStatus.SoftLiquidated; // or is fully soft liquidated
125
- return CrvUSDStatus.Nonexistant;
126
- };
127
-
128
- export const getCrvUsdAccountBalances = async (web3: Web3, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress, controllerAddress: EthAddress): Promise<PositionBalances> => {
129
- let balances: PositionBalances = {
130
- collateral: {},
131
- debt: {},
132
- };
133
-
134
- if (!address) {
135
- return balances;
136
- }
137
-
138
- const contract = CrvUSDViewContract(web3, network, block);
139
- const selectedMarket = Object.values(CrvUsdMarkets(network)).find(i => i.controllerAddress.toLowerCase() === controllerAddress.toLowerCase()) as CrvUSDMarketData;
140
-
141
- const data = await contract.methods.userData(selectedMarket.controllerAddress, address).call({}, block);
142
-
143
- balances = {
144
- collateral: {
145
- [addressMapping ? getAssetInfo(wethToEth(selectedMarket.collAsset), network).address.toLowerCase() : wethToEth(selectedMarket.collAsset)]: data.marketCollateralAmount,
146
- },
147
- debt: {
148
- [addressMapping ? getAssetInfo(wethToEth(selectedMarket.baseAsset), network).address.toLowerCase() : wethToEth(selectedMarket.baseAsset)]: data.debtAmount,
149
- },
150
- };
151
-
152
- return balances;
153
- };
154
-
155
- export const getCurveUsdUserData = async (web3: Web3, network: NetworkNumber, address: string, selectedMarket: CrvUSDMarketData, activeBand: string): Promise<CrvUSDUserData> => {
156
- const contract = CrvUSDViewContract(web3, network);
157
-
158
- const data = await contract.methods.userData(selectedMarket.controllerAddress, address).call();
159
- const collAsset = selectedMarket.collAsset;
160
- const debtAsset = selectedMarket.baseAsset;
161
-
162
- const health = assetAmountInEth(data.health);
163
- const healthPercent = new Dec(health).mul(100).toString();
164
- const collPrice = assetAmountInEth(data.collateralPrice, debtAsset);
165
- const collSupplied = assetAmountInEth(data.marketCollateralAmount, collAsset);
166
- const collSuppliedUsd = new Dec(collSupplied).mul(collPrice).toString();
167
- const crvUSDSupplied = assetAmountInEth(data.curveUsdCollateralAmount, debtAsset);
168
- const debtBorrowed = assetAmountInEth(data.debtAmount, debtAsset);
169
- const usedAssets: CrvUSDUsedAssets = data.loanExists ? {
170
- [collAsset]: {
171
- isSupplied: true,
172
- supplied: collSupplied,
173
- suppliedUsd: collSuppliedUsd, // need oracle price, or amm price
174
- borrowed: '0',
175
- borrowedUsd: '0',
176
- isBorrowed: false,
177
- symbol: collAsset,
178
- collateral: true,
179
- price: collPrice, // price_amm
180
- },
181
- [debtAsset]: {
182
- isSupplied: new Dec(crvUSDSupplied).gt('0'),
183
- collateral: new Dec(crvUSDSupplied).gt('0'),
184
- supplied: crvUSDSupplied,
185
- suppliedUsd: crvUSDSupplied,
186
- borrowed: debtBorrowed,
187
- borrowedUsd: debtBorrowed,
188
- isBorrowed: new Dec(debtBorrowed).gt('0'),
189
- symbol: 'crvUSD',
190
- price: '1',
191
- interestRate: '0',
192
- },
193
- } : {};
194
-
195
- const priceHigh = assetAmountInEth(data.priceHigh);
196
- const priceLow = assetAmountInEth(data.priceLow);
197
-
198
- const _userBands = data.loanExists ? (await getAndFormatBands(web3, network, selectedMarket, data.bandRange[0], data.bandRange[1])) : [];
199
-
200
- const status = data.loanExists ? getStatusForUser(data.bandRange, activeBand, crvUSDSupplied, collSupplied, healthPercent) : CrvUSDStatus.Nonexistant;
201
-
202
- const userBands = _userBands.map((band, index) => ({
203
- ...band,
204
- userDebtAmount: assetAmountInEth(data.usersBands[0][index], debtAsset),
205
- userCollAmount: assetAmountInEth(data.usersBands[1][index], collAsset),
206
- })).sort((a, b) => parseInt(b.id, 10) - parseInt(a.id, 10));
207
-
208
- return {
209
- ...data,
210
- debtAmount: assetAmountInEth(data.debtAmount, debtAsset),
211
- health,
212
- healthPercent,
213
- priceHigh,
214
- priceLow,
215
- liquidationDiscount: assetAmountInEth(data.liquidationDiscount),
216
- numOfBands: data.N,
217
- usedAssets,
218
- status,
219
- ...getCrvUsdAggregatedData({
220
- loanExists: data.loanExists, usedAssets, network: NetworkNumber.Eth, selectedMarket, numOfBands: data.N,
221
- }),
222
- userBands,
223
- };
224
- };
225
-
226
- export const getCurveUsdFullPositionData = async (web3: Web3, network: NetworkNumber, address: string, selectedMarket: CrvUSDMarketData): Promise<CrvUSDUserData> => {
227
- const marketData = await getCurveUsdGlobalData(web3, network, selectedMarket);
228
- const positionData = await getCurveUsdUserData(web3, network, address, selectedMarket, marketData.activeBand);
229
- return positionData;
1
+ import Dec from 'decimal.js';
2
+ import { assetAmountInEth, getAssetInfo } from '@defisaver/tokens';
3
+ import Web3 from 'web3';
4
+ import {
5
+ BandData, CrvUSDGlobalMarketData, CrvUSDMarketData, CrvUSDStatus, CrvUSDUsedAssets, CrvUSDUserData, CrvUSDVersions,
6
+ } from '../types';
7
+ import { multicall } from '../multicall';
8
+ import {
9
+ Blockish, EthAddress, NetworkNumber, PositionBalances,
10
+ } from '../types/common';
11
+ import { CrvUSDFactoryContract, CrvUSDViewContract } from '../contracts';
12
+ import { getCrvUsdAggregatedData } from '../helpers/curveUsdHelpers';
13
+ import { CrvUsdMarkets } from '../markets';
14
+ import { wethToEth } from '../services/utils';
15
+
16
+ const getAndFormatBands = async (web3: Web3, network: NetworkNumber, selectedMarket: CrvUSDMarketData, _minBand: string, _maxBand: string) => {
17
+ const contract = CrvUSDViewContract(web3, network);
18
+ const minBand = parseInt(_minBand, 10);
19
+ const maxBand = parseInt(_maxBand, 10);
20
+ const pivots: number[] = [];
21
+
22
+ // 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
23
+ let i = minBand;
24
+ while (i < maxBand) {
25
+ i += 200;
26
+ if (i > maxBand) {
27
+ pivots.push(maxBand);
28
+ } else {
29
+ pivots.push(i);
30
+ }
31
+ }
32
+
33
+ const bandsData = (await Promise.all(pivots.map(async (pivot, index) => {
34
+ let start = 0;
35
+ if (index === 0) {
36
+ start = minBand;
37
+ } else {
38
+ start = pivots[index - 1] + 1;
39
+ }
40
+ // @ts-ignore
41
+ const pivotedBandsData = await contract.methods.getBandsData(selectedMarket.controllerAddress, start, pivot).call();
42
+ return pivotedBandsData;
43
+ }))).flat();
44
+
45
+ return bandsData.map((band: BandData) => ({
46
+ id: band.id,
47
+ collAmount: assetAmountInEth(band.collAmount),
48
+ debtAmount: assetAmountInEth(band.debtAmount),
49
+ lowPrice: assetAmountInEth(band.lowPrice),
50
+ highPrice: assetAmountInEth(band.highPrice),
51
+ }));
52
+ };
53
+
54
+ export const getCurveUsdGlobalData = async (web3: Web3, network: NetworkNumber, selectedMarket: CrvUSDMarketData): Promise<CrvUSDGlobalMarketData> => {
55
+ const contract = CrvUSDViewContract(web3, network);
56
+ const factoryContract = CrvUSDFactoryContract(web3, network);
57
+ const collAsset = selectedMarket.collAsset;
58
+ const debtAsset = selectedMarket.baseAsset;
59
+
60
+ const multicallData = [
61
+ {
62
+ target: factoryContract.options.address,
63
+ abiItem: factoryContract.options.jsonInterface.find(({ name }) => name === 'debt_ceiling'),
64
+ params: [selectedMarket.controllerAddress],
65
+ },
66
+ {
67
+ target: factoryContract.options.address,
68
+ abiItem: factoryContract.options.jsonInterface.find(({ name }) => name === 'total_debt'),
69
+ params: [],
70
+ },
71
+ {
72
+ target: contract.options.address,
73
+ abiItem: contract.options.jsonInterface.find(({ name }) => name === 'globalData'),
74
+ params: [selectedMarket.controllerAddress],
75
+ },
76
+ ];
77
+ const multiRes = await multicall(multicallData, web3, network);
78
+ const data = multiRes[2][0];
79
+ const debtCeiling = assetAmountInEth(multiRes[0][0], debtAsset);
80
+
81
+ // all prices are in 18 decimals
82
+ const totalDebt = assetAmountInEth(data.totalDebt, debtAsset);
83
+ const ammPrice = assetAmountInEth(data.ammPrice, debtAsset);
84
+
85
+ const rate = assetAmountInEth(data.ammRate);
86
+ const futureRate = assetAmountInEth(data.monetaryPolicyRate);
87
+
88
+ const exponentRate = new Dec(rate).mul(365).mul(86400);
89
+ const exponentFutureRate = new Dec(futureRate).mul(365).mul(86400);
90
+ const borrowRate = new Dec(new Dec(2.718281828459).pow(exponentRate).minus(1)).mul(100)
91
+ .toString();
92
+ const futureBorrowRate = new Dec(new Dec(2.718281828459).pow(exponentFutureRate).minus(1)).mul(100)
93
+ .toString();
94
+
95
+ const bandsData = await getAndFormatBands(web3, network, selectedMarket, data.minBand, data.maxBand);
96
+
97
+ const leftToBorrow = new Dec(debtCeiling).minus(totalDebt).toString();
98
+ return {
99
+ ...data,
100
+ debtCeiling,
101
+ totalDebt,
102
+ ammPrice,
103
+ oraclePrice: assetAmountInEth(data.oraclePrice, debtAsset),
104
+ basePrice: assetAmountInEth(data.basePrice, debtAsset),
105
+ minted: assetAmountInEth(data.minted, debtAsset),
106
+ redeemed: assetAmountInEth(data.redeemed, debtAsset),
107
+ borrowRate,
108
+ futureBorrowRate,
109
+ bands: bandsData,
110
+ leftToBorrow,
111
+ };
112
+ };
113
+
114
+ const getStatusForUser = (bandRange: string[], activeBand: string, crvUSDSupplied: string, collSupplied: string, healthPercent: string) => {
115
+ // if bands are equal, that can only be [0,0] which means user doesn't have loan (min number of bands is 4)
116
+ if (new Dec(bandRange[0]).eq(bandRange[1])) return CrvUSDStatus.Nonexistant;
117
+ // if user doesn't have crvUSD as collateral, then his position is not in soft liquidation
118
+ if (new Dec(crvUSDSupplied).lte(0)) {
119
+ const isHealthRisky = new Dec(healthPercent).lt(10);
120
+ if (new Dec(bandRange[0]).minus(activeBand).lte(3) || isHealthRisky) return CrvUSDStatus.Risk; // if user band is less than 3 bands away from active band, his position is at risk
121
+ return CrvUSDStatus.Safe;
122
+ }
123
+ if (new Dec(bandRange[0]).lte(activeBand) && new Dec(bandRange[1]).gte(activeBand)) return CrvUSDStatus.SoftLiquidating; // user has crvUSD as coll so he is in soft liquidation
124
+ if (new Dec(collSupplied).lte(0) || new Dec(bandRange[1]).lte(activeBand)) return CrvUSDStatus.SoftLiquidated; // or is fully soft liquidated
125
+ return CrvUSDStatus.Nonexistant;
126
+ };
127
+
128
+ export const getCrvUsdAccountBalances = async (web3: Web3, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress, controllerAddress: EthAddress): Promise<PositionBalances> => {
129
+ let balances: PositionBalances = {
130
+ collateral: {},
131
+ debt: {},
132
+ };
133
+
134
+ if (!address) {
135
+ return balances;
136
+ }
137
+
138
+ const contract = CrvUSDViewContract(web3, network, block);
139
+ const selectedMarket = Object.values(CrvUsdMarkets(network)).find(i => i.controllerAddress.toLowerCase() === controllerAddress.toLowerCase()) as CrvUSDMarketData;
140
+
141
+ const data = await contract.methods.userData(selectedMarket.controllerAddress, address).call({}, block);
142
+
143
+ balances = {
144
+ collateral: {
145
+ [addressMapping ? getAssetInfo(wethToEth(selectedMarket.collAsset), network).address.toLowerCase() : wethToEth(selectedMarket.collAsset)]: data.marketCollateralAmount,
146
+ },
147
+ debt: {
148
+ [addressMapping ? getAssetInfo(wethToEth(selectedMarket.baseAsset), network).address.toLowerCase() : wethToEth(selectedMarket.baseAsset)]: data.debtAmount,
149
+ },
150
+ };
151
+
152
+ return balances;
153
+ };
154
+
155
+ export const getCurveUsdUserData = async (web3: Web3, network: NetworkNumber, address: string, selectedMarket: CrvUSDMarketData, activeBand: string): Promise<CrvUSDUserData> => {
156
+ const contract = CrvUSDViewContract(web3, network);
157
+
158
+ const data = await contract.methods.userData(selectedMarket.controllerAddress, address).call();
159
+ const collAsset = selectedMarket.collAsset;
160
+ const debtAsset = selectedMarket.baseAsset;
161
+
162
+ const health = assetAmountInEth(data.health);
163
+ const healthPercent = new Dec(health).mul(100).toString();
164
+ const collPrice = assetAmountInEth(data.collateralPrice, debtAsset);
165
+ const collSupplied = assetAmountInEth(data.marketCollateralAmount, collAsset);
166
+ const collSuppliedUsd = new Dec(collSupplied).mul(collPrice).toString();
167
+ const crvUSDSupplied = assetAmountInEth(data.curveUsdCollateralAmount, debtAsset);
168
+ const debtBorrowed = assetAmountInEth(data.debtAmount, debtAsset);
169
+ const usedAssets: CrvUSDUsedAssets = data.loanExists ? {
170
+ [collAsset]: {
171
+ isSupplied: true,
172
+ supplied: collSupplied,
173
+ suppliedUsd: collSuppliedUsd, // need oracle price, or amm price
174
+ borrowed: '0',
175
+ borrowedUsd: '0',
176
+ isBorrowed: false,
177
+ symbol: collAsset,
178
+ collateral: true,
179
+ price: collPrice, // price_amm
180
+ },
181
+ [debtAsset]: {
182
+ isSupplied: new Dec(crvUSDSupplied).gt('0'),
183
+ collateral: new Dec(crvUSDSupplied).gt('0'),
184
+ supplied: crvUSDSupplied,
185
+ suppliedUsd: crvUSDSupplied,
186
+ borrowed: debtBorrowed,
187
+ borrowedUsd: debtBorrowed,
188
+ isBorrowed: new Dec(debtBorrowed).gt('0'),
189
+ symbol: 'crvUSD',
190
+ price: '1',
191
+ interestRate: '0',
192
+ },
193
+ } : {};
194
+
195
+ const priceHigh = assetAmountInEth(data.priceHigh);
196
+ const priceLow = assetAmountInEth(data.priceLow);
197
+
198
+ const _userBands = data.loanExists ? (await getAndFormatBands(web3, network, selectedMarket, data.bandRange[0], data.bandRange[1])) : [];
199
+
200
+ const status = data.loanExists ? getStatusForUser(data.bandRange, activeBand, crvUSDSupplied, collSupplied, healthPercent) : CrvUSDStatus.Nonexistant;
201
+
202
+ const userBands = _userBands.map((band, index) => ({
203
+ ...band,
204
+ userDebtAmount: assetAmountInEth(data.usersBands[0][index], debtAsset),
205
+ userCollAmount: assetAmountInEth(data.usersBands[1][index], collAsset),
206
+ })).sort((a, b) => parseInt(b.id, 10) - parseInt(a.id, 10));
207
+
208
+ return {
209
+ ...data,
210
+ debtAmount: assetAmountInEth(data.debtAmount, debtAsset),
211
+ health,
212
+ healthPercent,
213
+ priceHigh,
214
+ priceLow,
215
+ liquidationDiscount: assetAmountInEth(data.liquidationDiscount),
216
+ numOfBands: data.N,
217
+ usedAssets,
218
+ status,
219
+ ...getCrvUsdAggregatedData({
220
+ loanExists: data.loanExists, usedAssets, network: NetworkNumber.Eth, selectedMarket, numOfBands: data.N,
221
+ }),
222
+ userBands,
223
+ };
224
+ };
225
+
226
+ export const getCurveUsdFullPositionData = async (web3: Web3, network: NetworkNumber, address: string, selectedMarket: CrvUSDMarketData): Promise<CrvUSDUserData> => {
227
+ const marketData = await getCurveUsdGlobalData(web3, network, selectedMarket);
228
+ const positionData = await getCurveUsdUserData(web3, network, address, selectedMarket, marketData.activeBand);
229
+ return positionData;
230
230
  };