@defisaver/positions-sdk 1.0.3 → 1.0.5

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 (108) hide show
  1. package/.mocharc.json +4 -4
  2. package/.nvmrc +1 -1
  3. package/README.md +69 -69
  4. package/cjs/aaveV3/index.js +7 -7
  5. package/cjs/curveUsd/index.js +8 -1
  6. package/cjs/fluid/index.d.ts +2 -0
  7. package/cjs/helpers/aaveHelpers/index.js +6 -0
  8. package/cjs/helpers/compoundHelpers/index.js +2 -0
  9. package/cjs/helpers/eulerHelpers/index.js +2 -0
  10. package/cjs/helpers/fluidHelpers/index.js +2 -0
  11. package/cjs/helpers/morphoBlueHelpers/index.js +68 -66
  12. package/cjs/helpers/sparkHelpers/index.js +6 -0
  13. package/cjs/staking/staking.d.ts +2 -1
  14. package/cjs/staking/staking.js +3 -7
  15. package/cjs/types/aave.d.ts +4 -0
  16. package/cjs/types/compound.d.ts +2 -0
  17. package/cjs/types/curveUsd.d.ts +1 -0
  18. package/cjs/types/euler.d.ts +2 -0
  19. package/cjs/types/fluid.d.ts +2 -0
  20. package/cjs/types/morphoBlue.d.ts +2 -0
  21. package/cjs/types/spark.d.ts +4 -0
  22. package/esm/aaveV3/index.js +7 -7
  23. package/esm/curveUsd/index.js +10 -3
  24. package/esm/fluid/index.d.ts +2 -0
  25. package/esm/helpers/aaveHelpers/index.js +6 -0
  26. package/esm/helpers/compoundHelpers/index.js +2 -0
  27. package/esm/helpers/eulerHelpers/index.js +2 -0
  28. package/esm/helpers/fluidHelpers/index.js +2 -0
  29. package/esm/helpers/morphoBlueHelpers/index.js +68 -66
  30. package/esm/helpers/sparkHelpers/index.js +6 -0
  31. package/esm/staking/staking.d.ts +2 -1
  32. package/esm/staking/staking.js +3 -6
  33. package/esm/types/aave.d.ts +4 -0
  34. package/esm/types/compound.d.ts +2 -0
  35. package/esm/types/curveUsd.d.ts +1 -0
  36. package/esm/types/euler.d.ts +2 -0
  37. package/esm/types/fluid.d.ts +2 -0
  38. package/esm/types/morphoBlue.d.ts +2 -0
  39. package/esm/types/spark.d.ts +4 -0
  40. package/package.json +54 -52
  41. package/src/aaveV2/index.ts +227 -227
  42. package/src/aaveV3/index.ts +624 -625
  43. package/src/assets/index.ts +60 -60
  44. package/src/chickenBonds/index.ts +123 -123
  45. package/src/compoundV2/index.ts +220 -220
  46. package/src/compoundV3/index.ts +291 -291
  47. package/src/config/contracts.js +1147 -1147
  48. package/src/constants/index.ts +6 -6
  49. package/src/contracts.ts +134 -134
  50. package/src/curveUsd/index.ts +239 -230
  51. package/src/eulerV2/index.ts +303 -303
  52. package/src/exchange/index.ts +17 -17
  53. package/src/fluid/index.ts +354 -354
  54. package/src/helpers/aaveHelpers/index.ts +204 -198
  55. package/src/helpers/chickenBondsHelpers/index.ts +23 -23
  56. package/src/helpers/compoundHelpers/index.ts +248 -246
  57. package/src/helpers/curveUsdHelpers/index.ts +40 -40
  58. package/src/helpers/eulerHelpers/index.ts +234 -232
  59. package/src/helpers/fluidHelpers/index.ts +56 -53
  60. package/src/helpers/index.ts +11 -11
  61. package/src/helpers/liquityV2Helpers/index.ts +80 -80
  62. package/src/helpers/llamaLendHelpers/index.ts +53 -53
  63. package/src/helpers/makerHelpers/index.ts +94 -94
  64. package/src/helpers/morphoBlueHelpers/index.ts +367 -365
  65. package/src/helpers/sparkHelpers/index.ts +155 -150
  66. package/src/index.ts +52 -52
  67. package/src/liquity/index.ts +116 -116
  68. package/src/liquityV2/index.ts +295 -295
  69. package/src/llamaLend/index.ts +275 -275
  70. package/src/maker/index.ts +117 -117
  71. package/src/markets/aave/index.ts +152 -152
  72. package/src/markets/aave/marketAssets.ts +44 -44
  73. package/src/markets/compound/index.ts +213 -213
  74. package/src/markets/compound/marketsAssets.ts +82 -82
  75. package/src/markets/curveUsd/index.ts +69 -69
  76. package/src/markets/euler/index.ts +26 -26
  77. package/src/markets/fluid/index.ts +2012 -2012
  78. package/src/markets/index.ts +27 -27
  79. package/src/markets/liquityV2/index.ts +54 -54
  80. package/src/markets/llamaLend/contractAddresses.ts +141 -141
  81. package/src/markets/llamaLend/index.ts +235 -235
  82. package/src/markets/morphoBlue/index.ts +895 -895
  83. package/src/markets/spark/index.ts +29 -29
  84. package/src/markets/spark/marketAssets.ts +10 -10
  85. package/src/moneymarket/moneymarketCommonService.ts +80 -80
  86. package/src/morphoAaveV2/index.ts +256 -256
  87. package/src/morphoAaveV3/index.ts +630 -630
  88. package/src/morphoBlue/index.ts +202 -202
  89. package/src/multicall/index.ts +33 -33
  90. package/src/services/priceService.ts +91 -91
  91. package/src/services/utils.ts +59 -59
  92. package/src/setup.ts +8 -8
  93. package/src/spark/index.ts +460 -460
  94. package/src/staking/staking.ts +217 -220
  95. package/src/types/aave.ts +275 -271
  96. package/src/types/chickenBonds.ts +45 -45
  97. package/src/types/common.ts +84 -84
  98. package/src/types/compound.ts +133 -131
  99. package/src/types/curveUsd.ts +119 -118
  100. package/src/types/euler.ts +173 -171
  101. package/src/types/fluid.ts +268 -266
  102. package/src/types/index.ts +11 -11
  103. package/src/types/liquity.ts +30 -30
  104. package/src/types/liquityV2.ts +119 -119
  105. package/src/types/llamaLend.ts +155 -155
  106. package/src/types/maker.ts +50 -50
  107. package/src/types/morphoBlue.ts +194 -192
  108. package/src/types/spark.ts +135 -131
@@ -1,355 +1,355 @@
1
- import Web3 from 'web3';
2
- import Dec from 'decimal.js';
3
- import { assetAmountInEth, getAssetInfo, getAssetInfoByAddress } from '@defisaver/tokens';
4
- import { EthAddress, NetworkNumber } from '../types/common';
5
- import {
6
- FluidAggregatedVaultData,
7
- FluidAssetData, FluidAssetsData,
8
- FluidMarketData,
9
- FluidMarketInfo,
10
- FluidUsedAsset,
11
- FluidUsedAssets,
12
- FluidVaultData,
13
- FluidVaultType, InnerFluidMarketData,
14
- } from '../types';
15
- import { DFSFeedRegistryContract, FeedRegistryContract, FluidViewContract } from '../contracts';
16
- import { getEthAmountForDecimals, isMainnetNetwork } from '../services/utils';
17
- import { getFluidAggregatedData } from '../helpers/fluidHelpers';
18
- import { FluidView } from '../types/contracts/generated';
19
- import { chunkAndMulticall } from '../multicall';
20
- import { getFluidMarketInfoById, getFluidVersionsDataForNetwork, getFTokenAddress } from '../markets';
21
- import { USD_QUOTE } from '../constants';
22
- import { getChainlinkAssetAddress, getWstETHPriceFluid } from '../services/priceService';
23
- import { getStakingApy, STAKING_ASSETS } from '../staking';
24
-
25
- export const EMPTY_USED_ASSET = {
26
- isSupplied: false,
27
- isBorrowed: false,
28
- supplied: '0',
29
- suppliedUsd: '0',
30
- borrowed: '0',
31
- borrowedUsd: '0',
32
- symbol: '',
33
- collateral: false,
34
- };
35
-
36
- const parseVaultType = (vaultType: number) => {
37
- switch (vaultType) {
38
- case 10000: return FluidVaultType.T1;
39
- case 20000: return FluidVaultType.T2;
40
- case 30000: return FluidVaultType.T3;
41
- case 40000: return FluidVaultType.T4;
42
- default: return FluidVaultType.Unknown;
43
- }
44
- };
45
-
46
- const parseMarketData = async (web3: Web3, data: FluidView.VaultDataStructOutputStruct, network: NetworkNumber, mainnetWeb3: Web3) => {
47
- const collAsset = getAssetInfoByAddress(data.supplyToken0, network);
48
- const debtAsset = getAssetInfoByAddress(data.borrowToken0, network);
49
-
50
- const supplyRate = new Dec(data.supplyRateVault).div(100).toString();
51
- const borrowRate = new Dec(data.borrowRateVault).div(100).toString();
52
-
53
- const oracleScaleFactor = new Dec(27).add(debtAsset.decimals).sub(collAsset.decimals).toString();
54
- const oracleScale = new Dec(10).pow(oracleScaleFactor).toString();
55
- const oraclePrice = new Dec(data.oraclePriceOperate).div(oracleScale).toString();
56
-
57
- const isTokenUSDA = debtAsset.symbol === 'USDA';
58
- const isMainnet = isMainnetNetwork(network);
59
- const loanTokenFeedAddress = getChainlinkAssetAddress(debtAsset.symbol, network);
60
-
61
- let loanTokenPrice;
62
- if (debtAsset.symbol === 'wstETH') {
63
- // need to handle wstETH for l2s inside getWstETHPrice
64
- loanTokenPrice = await getWstETHPriceFluid(web3, network);
65
- } else if (isMainnet) {
66
- const feedRegistryContract = FeedRegistryContract(web3, NetworkNumber.Eth);
67
- loanTokenPrice = isTokenUSDA ? '100000000' : await feedRegistryContract.methods.latestAnswer(loanTokenFeedAddress, USD_QUOTE).call();
68
- } else {
69
- // Currently only base network is supported
70
- const feedRegistryContract = DFSFeedRegistryContract(web3, network);
71
- const roundPriceData = isTokenUSDA ? { answer: '100000000' } : await feedRegistryContract.methods.latestRoundData(loanTokenFeedAddress, USD_QUOTE).call();
72
- loanTokenPrice = roundPriceData.answer;
73
- }
74
-
75
- const debtPriceParsed = new Dec(loanTokenPrice).div(1e8).toString();
76
-
77
- const collAssetData: FluidAssetData = {
78
- symbol: collAsset.symbol,
79
- address: collAsset.address,
80
- price: new Dec(debtPriceParsed).mul(oraclePrice).toString(),
81
- totalSupply: data.totalSupplyVault,
82
- totalBorrow: data.totalBorrowVault,
83
- canBeSupplied: true,
84
- canBeBorrowed: false,
85
- supplyRate,
86
- borrowRate: '0',
87
- };
88
-
89
- if (STAKING_ASSETS.includes(collAsset.symbol)) {
90
- collAssetData.incentiveSupplyApy = await getStakingApy(collAsset.symbol, mainnetWeb3);
91
- collAssetData.incentiveSupplyToken = collAsset.symbol;
92
- }
93
-
94
- const debtAssetData: FluidAssetData = {
95
- symbol: debtAsset.symbol,
96
- address: debtAsset.address,
97
- price: debtPriceParsed,
98
- totalSupply: data.totalSupplyVault,
99
- totalBorrow: data.totalBorrowVault,
100
- canBeSupplied: false,
101
- canBeBorrowed: true,
102
- supplyRate: '0',
103
- borrowRate,
104
- };
105
-
106
- if (STAKING_ASSETS.includes(debtAssetData.symbol)) {
107
- debtAssetData.incentiveBorrowApy = await getStakingApy(debtAsset.symbol, mainnetWeb3);
108
- debtAssetData.incentiveBorrowToken = debtAsset.symbol;
109
- }
110
-
111
- const assetsData = {
112
- [collAsset.symbol]: collAssetData,
113
- [debtAsset.symbol]: debtAssetData,
114
- };
115
- const marketInfo = getFluidMarketInfoById(+data.vaultId, network);
116
- const totalSupplyVault = getEthAmountForDecimals(data.totalSupplyVault, collAsset.decimals);
117
- const totalBorrowVault = getEthAmountForDecimals(data.totalBorrowVault, debtAsset.decimals);
118
-
119
- const liqRatio = new Dec(data.liquidationThreshold).div(100).toString();
120
- const liquidationMaxLimit = new Dec(data.liquidationMaxLimit).div(100).toString();
121
- const liqFactor = new Dec(data.liquidationThreshold).div(10_000).toString();
122
-
123
- const marketData = {
124
- vaultId: +data.vaultId,
125
- vaultValue: marketInfo?.value,
126
- isSmartColl: data.isSmartColl,
127
- isSmartDebt: data.isSmartDebt,
128
- marketAddress: data.vault,
129
- vaultType: parseVaultType(+data.vaultType),
130
- oracle: data.oracle,
131
- liquidationPenaltyPercent: new Dec(data.liquidationPenalty).div(100).toString(),
132
- collFactor: new Dec(data.collateralFactor).div(10000).toString(), // we want actual factor, not in %, so we divide by 10000 instead of 100
133
- liquidationRatio: liqRatio,
134
- liqFactor,
135
- minRatio: new Dec(1).div(liqFactor).mul(100).toString(),
136
- collAsset0: collAsset.symbol,
137
- debtAsset0: debtAsset.symbol,
138
- totalPositions: data.totalPositions,
139
- totalSupplyVault,
140
- totalBorrowVault,
141
- totalSupplyVaultUsd: new Dec(totalSupplyVault).mul(collAssetData.price).toString(),
142
- totalBorrowVaultUsd: new Dec(totalBorrowVault).mul(debtAssetData.price).toString(),
143
- withdrawalLimit: getEthAmountForDecimals(data.withdrawalLimit, collAsset.decimals),
144
- withdrawableUntilLimit: getEthAmountForDecimals(data.withdrawableUntilLimit, collAsset.decimals),
145
- withdrawable: getEthAmountForDecimals(data.withdrawable, collAsset.decimals),
146
- borrowLimit: getEthAmountForDecimals(data.borrowLimit, debtAsset.decimals),
147
- borrowableUntilLimit: getEthAmountForDecimals(data.borrowableUntilLimit, debtAsset.decimals),
148
- borrowable: getEthAmountForDecimals(data.borrowable, debtAsset.decimals),
149
- borrowLimitUtilization: getEthAmountForDecimals(data.borrowLimitUtilization, debtAsset.decimals),
150
- maxBorrowLimit: getEthAmountForDecimals(data.maxBorrowLimit, debtAsset.decimals),
151
- baseBorrowLimit: getEthAmountForDecimals(data.baseBorrowLimit, debtAsset.decimals),
152
- minimumBorrowing: getEthAmountForDecimals(data.minimumBorrowing, debtAsset.decimals),
153
- liquidationMaxLimit,
154
- borrowRate,
155
- supplyRate,
156
- };
157
-
158
- return {
159
- assetsData,
160
- marketData,
161
- } as FluidMarketData;
162
- };
163
-
164
- export const EMPTY_FLUID_DATA = {
165
- usedAssets: {},
166
- suppliedUsd: '0',
167
- borrowedUsd: '0',
168
- borrowLimitUsd: '0',
169
- leftToBorrowUsd: '0',
170
- ratio: '0',
171
- minRatio: '0',
172
- netApy: '0',
173
- incentiveUsd: '0',
174
- totalInterestUsd: '0',
175
- isSubscribedToAutomation: false,
176
- automationResubscribeRequired: false,
177
- lastUpdated: Date.now(),
178
- };
179
-
180
- const parseUserData = (userPositionData: FluidView.UserPositionStructOutputStruct, vaultData: FluidMarketData): FluidVaultData => {
181
- const {
182
- assetsData,
183
- marketData,
184
- } = vaultData;
185
-
186
- const payload = {
187
- owner: userPositionData.owner,
188
- vaultId: marketData.vaultId,
189
- ...EMPTY_FLUID_DATA,
190
- lastUpdated: Date.now(),
191
- };
192
- const collAsset = getAssetInfo(marketData.collAsset0);
193
- const debtAsset = getAssetInfo(marketData.debtAsset0);
194
-
195
- const supplied = getEthAmountForDecimals(userPositionData.supply, collAsset.decimals);
196
- const borrowed = getEthAmountForDecimals(userPositionData.borrow, debtAsset.decimals);
197
-
198
- const collUsedAsset: FluidUsedAsset = {
199
- ...EMPTY_USED_ASSET,
200
- symbol: collAsset.symbol,
201
- collateral: true,
202
- supplied,
203
- suppliedUsd: new Dec(supplied).mul(assetsData[collAsset.symbol].price).toString(),
204
- isSupplied: new Dec(supplied).gt(0),
205
- };
206
-
207
- const debtUsedAsset: FluidUsedAsset = {
208
- ...EMPTY_USED_ASSET,
209
- symbol: debtAsset.symbol,
210
- collateral: false,
211
- borrowed,
212
- borrowedUsd: new Dec(borrowed).mul(assetsData[debtAsset.symbol].price).toString(),
213
- isBorrowed: new Dec(borrowed).gt(0),
214
- };
215
-
216
- const usedAssets: FluidUsedAssets = {
217
- [collAsset.symbol]: collUsedAsset,
218
- [debtAsset.symbol]: debtUsedAsset,
219
- };
220
-
221
- return {
222
- ...payload,
223
- usedAssets,
224
- ...(getFluidAggregatedData({
225
- usedAssets,
226
- assetsData,
227
- marketData,
228
- }) as FluidAggregatedVaultData),
229
- };
230
- };
231
-
232
- export const getFluidMarketData = async (web3: Web3, network: NetworkNumber, market: FluidMarketInfo, mainnetWeb3: Web3) => {
233
- const view = FluidViewContract(web3, network);
234
-
235
- const data = await view.methods.getVaultData(market.marketAddress).call();
236
-
237
- return parseMarketData(web3, data, network, mainnetWeb3);
238
- };
239
-
240
- export const getFluidVaultIdsForUser = async (web3: Web3,
241
- network:NetworkNumber,
242
- user: EthAddress): Promise<string[]> => {
243
- const view = FluidViewContract(web3, network);
244
-
245
- return view.methods.getUserNftIds(user).call();
246
- };
247
-
248
-
249
- export const getFluidPosition = async (
250
- web3: Web3,
251
- network: NetworkNumber,
252
- vaultId: string,
253
- extractedState: {
254
- assetsData: FluidAssetsData
255
- marketData: InnerFluidMarketData,
256
- },
257
- ): Promise<FluidVaultData> => {
258
- const view = FluidViewContract(web3, network);
259
-
260
- const data = await view.methods.getPositionByNftId(vaultId).call();
261
-
262
- const userPositionData = data[0];
263
-
264
- return parseUserData(userPositionData, extractedState);
265
- };
266
-
267
- export const getFluidPositionWithMarket = async (web3: Web3, network: NetworkNumber, vaultId: string, mainnetWeb3: Web3) => {
268
- const view = FluidViewContract(web3, network);
269
- const data = await view.methods.getPositionByNftId(vaultId).call();
270
- const marketData = await parseMarketData(web3, data.vault, network, mainnetWeb3);
271
- const userData = parseUserData(data.position, marketData);
272
-
273
- return {
274
- userData,
275
- marketData,
276
- };
277
- };
278
-
279
- export const getAllFluidMarketDataChunked = async (network: NetworkNumber, web3: Web3, mainnetWeb3: Web3) => {
280
- const versions = getFluidVersionsDataForNetwork(network);
281
- const view = FluidViewContract(web3, network);
282
- const calls = versions.map((version) => ({
283
- target: view.options.address,
284
- abiItem: view.options.jsonInterface.find((item) => item.name === 'getVaultData'),
285
- params: [version.marketAddress],
286
- }));
287
-
288
- const data = await chunkAndMulticall(calls, 10, 'latest', web3, network);
289
- // @ts-ignore
290
- return Promise.all(data.map(async (item, i) => parseMarketData(web3, item.vaultData, network, mainnetWeb3)));
291
- };
292
-
293
- export const getFluidTokenData = async (web3: Web3, network: NetworkNumber, token: string) => {
294
- const view = FluidViewContract(web3, network);
295
- const fTokenAddress = getFTokenAddress(token, network);
296
- const data = await view.methods.getFTokenData(fTokenAddress).call();
297
- const supplyRate = new Dec(data.supplyRate).div(100).toString();
298
- const rewardsRate = new Dec(data.rewardsRate).div(1e12).toString();
299
- const decimals = data.decimals;
300
-
301
- const depositRate = new Dec(getEthAmountForDecimals(data.convertToShares, decimals)).toString();
302
- const withdrawRate = new Dec(getEthAmountForDecimals(data.convertToAssets, decimals)).toString();
303
-
304
- return {
305
- fTokenAddress,
306
- fTokenSymbol: data.symbol,
307
- decimals,
308
- totalDeposited: getEthAmountForDecimals(data.totalAssets, decimals),
309
- withdrawable: getEthAmountForDecimals(data.withdrawable, decimals),
310
- apy: new Dec(supplyRate).add(rewardsRate).toString(),
311
- depositRate,
312
- withdrawRate,
313
- };
314
- };
315
-
316
- export const getFluidDepositData = async (web3: Web3, network: NetworkNumber, token: string, address: EthAddress) => {
317
- const view = FluidViewContract(web3, network);
318
- const fTokenAddress = getFTokenAddress(token, network);
319
- const { fTokenData, userPosition } = await view.methods.getUserEarnPositionWithFToken(fTokenAddress, address).call();
320
-
321
- const supplyRate = new Dec(fTokenData.supplyRate).div(100).toString();
322
- const rewardsRate = new Dec(fTokenData.rewardsRate).div(1e12).toString();
323
- const decimals = fTokenData.decimals;
324
-
325
- const depositRate = new Dec(getEthAmountForDecimals(fTokenData.convertToShares, decimals)).toString();
326
- const withdrawRate = new Dec(getEthAmountForDecimals(fTokenData.convertToAssets, decimals)).toString();
327
-
328
- return {
329
- fTokenAddress,
330
- fTokenSymbol: fTokenData.symbol,
331
- decimals,
332
- totalDeposited: getEthAmountForDecimals(fTokenData.totalAssets, decimals),
333
- withdrawable: getEthAmountForDecimals(fTokenData.withdrawable, decimals),
334
- apy: new Dec(supplyRate).add(rewardsRate).toString(),
335
- depositRate,
336
- withdrawRate,
337
- deposited: getEthAmountForDecimals(userPosition.underlyingAssets, decimals),
338
- depositedShares: getEthAmountForDecimals(userPosition.fTokenShares, decimals),
339
- };
340
- };
341
-
342
- export const getUserPositions = async (web3: Web3, network: NetworkNumber, user: EthAddress, mainnetWeb3: Web3) => {
343
- const view = FluidViewContract(web3, network);
344
-
345
- const data = await view.methods.getUserPositions(user).call();
346
-
347
- const parsedMarketData = await Promise.all(data.vaults.map(async (vaultData) => parseMarketData(web3, vaultData, network, mainnetWeb3)));
348
-
349
- const userData = data.positions.map((position, i) => ({ ...parseUserData(position, parsedMarketData[i]), nftId: position.nftId }));
350
-
351
- return parsedMarketData.map((market, i) => ({
352
- marketData: market,
353
- userData: userData[i],
354
- }));
1
+ import Web3 from 'web3';
2
+ import Dec from 'decimal.js';
3
+ import { assetAmountInEth, getAssetInfo, getAssetInfoByAddress } from '@defisaver/tokens';
4
+ import { EthAddress, NetworkNumber } from '../types/common';
5
+ import {
6
+ FluidAggregatedVaultData,
7
+ FluidAssetData, FluidAssetsData,
8
+ FluidMarketData,
9
+ FluidMarketInfo,
10
+ FluidUsedAsset,
11
+ FluidUsedAssets,
12
+ FluidVaultData,
13
+ FluidVaultType, InnerFluidMarketData,
14
+ } from '../types';
15
+ import { DFSFeedRegistryContract, FeedRegistryContract, FluidViewContract } from '../contracts';
16
+ import { getEthAmountForDecimals, isMainnetNetwork } from '../services/utils';
17
+ import { getFluidAggregatedData } from '../helpers/fluidHelpers';
18
+ import { FluidView } from '../types/contracts/generated';
19
+ import { chunkAndMulticall } from '../multicall';
20
+ import { getFluidMarketInfoById, getFluidVersionsDataForNetwork, getFTokenAddress } from '../markets';
21
+ import { USD_QUOTE } from '../constants';
22
+ import { getChainlinkAssetAddress, getWstETHPriceFluid } from '../services/priceService';
23
+ import { getStakingApy, STAKING_ASSETS } from '../staking';
24
+
25
+ export const EMPTY_USED_ASSET = {
26
+ isSupplied: false,
27
+ isBorrowed: false,
28
+ supplied: '0',
29
+ suppliedUsd: '0',
30
+ borrowed: '0',
31
+ borrowedUsd: '0',
32
+ symbol: '',
33
+ collateral: false,
34
+ };
35
+
36
+ const parseVaultType = (vaultType: number) => {
37
+ switch (vaultType) {
38
+ case 10000: return FluidVaultType.T1;
39
+ case 20000: return FluidVaultType.T2;
40
+ case 30000: return FluidVaultType.T3;
41
+ case 40000: return FluidVaultType.T4;
42
+ default: return FluidVaultType.Unknown;
43
+ }
44
+ };
45
+
46
+ const parseMarketData = async (web3: Web3, data: FluidView.VaultDataStructOutputStruct, network: NetworkNumber, mainnetWeb3: Web3) => {
47
+ const collAsset = getAssetInfoByAddress(data.supplyToken0, network);
48
+ const debtAsset = getAssetInfoByAddress(data.borrowToken0, network);
49
+
50
+ const supplyRate = new Dec(data.supplyRateVault).div(100).toString();
51
+ const borrowRate = new Dec(data.borrowRateVault).div(100).toString();
52
+
53
+ const oracleScaleFactor = new Dec(27).add(debtAsset.decimals).sub(collAsset.decimals).toString();
54
+ const oracleScale = new Dec(10).pow(oracleScaleFactor).toString();
55
+ const oraclePrice = new Dec(data.oraclePriceOperate).div(oracleScale).toString();
56
+
57
+ const isTokenUSDA = debtAsset.symbol === 'USDA';
58
+ const isMainnet = isMainnetNetwork(network);
59
+ const loanTokenFeedAddress = getChainlinkAssetAddress(debtAsset.symbol, network);
60
+
61
+ let loanTokenPrice;
62
+ if (debtAsset.symbol === 'wstETH') {
63
+ // need to handle wstETH for l2s inside getWstETHPrice
64
+ loanTokenPrice = await getWstETHPriceFluid(web3, network);
65
+ } else if (isMainnet) {
66
+ const feedRegistryContract = FeedRegistryContract(web3, NetworkNumber.Eth);
67
+ loanTokenPrice = isTokenUSDA ? '100000000' : await feedRegistryContract.methods.latestAnswer(loanTokenFeedAddress, USD_QUOTE).call();
68
+ } else {
69
+ // Currently only base network is supported
70
+ const feedRegistryContract = DFSFeedRegistryContract(web3, network);
71
+ const roundPriceData = isTokenUSDA ? { answer: '100000000' } : await feedRegistryContract.methods.latestRoundData(loanTokenFeedAddress, USD_QUOTE).call();
72
+ loanTokenPrice = roundPriceData.answer;
73
+ }
74
+
75
+ const debtPriceParsed = new Dec(loanTokenPrice).div(1e8).toString();
76
+
77
+ const collAssetData: FluidAssetData = {
78
+ symbol: collAsset.symbol,
79
+ address: collAsset.address,
80
+ price: new Dec(debtPriceParsed).mul(oraclePrice).toString(),
81
+ totalSupply: data.totalSupplyVault,
82
+ totalBorrow: data.totalBorrowVault,
83
+ canBeSupplied: true,
84
+ canBeBorrowed: false,
85
+ supplyRate,
86
+ borrowRate: '0',
87
+ };
88
+
89
+ if (STAKING_ASSETS.includes(collAsset.symbol)) {
90
+ collAssetData.incentiveSupplyApy = await getStakingApy(collAsset.symbol, mainnetWeb3);
91
+ collAssetData.incentiveSupplyToken = collAsset.symbol;
92
+ }
93
+
94
+ const debtAssetData: FluidAssetData = {
95
+ symbol: debtAsset.symbol,
96
+ address: debtAsset.address,
97
+ price: debtPriceParsed,
98
+ totalSupply: data.totalSupplyVault,
99
+ totalBorrow: data.totalBorrowVault,
100
+ canBeSupplied: false,
101
+ canBeBorrowed: true,
102
+ supplyRate: '0',
103
+ borrowRate,
104
+ };
105
+
106
+ if (STAKING_ASSETS.includes(debtAssetData.symbol)) {
107
+ debtAssetData.incentiveBorrowApy = await getStakingApy(debtAsset.symbol, mainnetWeb3);
108
+ debtAssetData.incentiveBorrowToken = debtAsset.symbol;
109
+ }
110
+
111
+ const assetsData = {
112
+ [collAsset.symbol]: collAssetData,
113
+ [debtAsset.symbol]: debtAssetData,
114
+ };
115
+ const marketInfo = getFluidMarketInfoById(+data.vaultId, network);
116
+ const totalSupplyVault = getEthAmountForDecimals(data.totalSupplyVault, collAsset.decimals);
117
+ const totalBorrowVault = getEthAmountForDecimals(data.totalBorrowVault, debtAsset.decimals);
118
+
119
+ const liqRatio = new Dec(data.liquidationThreshold).div(100).toString();
120
+ const liquidationMaxLimit = new Dec(data.liquidationMaxLimit).div(100).toString();
121
+ const liqFactor = new Dec(data.liquidationThreshold).div(10_000).toString();
122
+
123
+ const marketData = {
124
+ vaultId: +data.vaultId,
125
+ vaultValue: marketInfo?.value,
126
+ isSmartColl: data.isSmartColl,
127
+ isSmartDebt: data.isSmartDebt,
128
+ marketAddress: data.vault,
129
+ vaultType: parseVaultType(+data.vaultType),
130
+ oracle: data.oracle,
131
+ liquidationPenaltyPercent: new Dec(data.liquidationPenalty).div(100).toString(),
132
+ collFactor: new Dec(data.collateralFactor).div(10000).toString(), // we want actual factor, not in %, so we divide by 10000 instead of 100
133
+ liquidationRatio: liqRatio,
134
+ liqFactor,
135
+ minRatio: new Dec(1).div(liqFactor).mul(100).toString(),
136
+ collAsset0: collAsset.symbol,
137
+ debtAsset0: debtAsset.symbol,
138
+ totalPositions: data.totalPositions,
139
+ totalSupplyVault,
140
+ totalBorrowVault,
141
+ totalSupplyVaultUsd: new Dec(totalSupplyVault).mul(collAssetData.price).toString(),
142
+ totalBorrowVaultUsd: new Dec(totalBorrowVault).mul(debtAssetData.price).toString(),
143
+ withdrawalLimit: getEthAmountForDecimals(data.withdrawalLimit, collAsset.decimals),
144
+ withdrawableUntilLimit: getEthAmountForDecimals(data.withdrawableUntilLimit, collAsset.decimals),
145
+ withdrawable: getEthAmountForDecimals(data.withdrawable, collAsset.decimals),
146
+ borrowLimit: getEthAmountForDecimals(data.borrowLimit, debtAsset.decimals),
147
+ borrowableUntilLimit: getEthAmountForDecimals(data.borrowableUntilLimit, debtAsset.decimals),
148
+ borrowable: getEthAmountForDecimals(data.borrowable, debtAsset.decimals),
149
+ borrowLimitUtilization: getEthAmountForDecimals(data.borrowLimitUtilization, debtAsset.decimals),
150
+ maxBorrowLimit: getEthAmountForDecimals(data.maxBorrowLimit, debtAsset.decimals),
151
+ baseBorrowLimit: getEthAmountForDecimals(data.baseBorrowLimit, debtAsset.decimals),
152
+ minimumBorrowing: getEthAmountForDecimals(data.minimumBorrowing, debtAsset.decimals),
153
+ liquidationMaxLimit,
154
+ borrowRate,
155
+ supplyRate,
156
+ };
157
+
158
+ return {
159
+ assetsData,
160
+ marketData,
161
+ } as FluidMarketData;
162
+ };
163
+
164
+ export const EMPTY_FLUID_DATA = {
165
+ usedAssets: {},
166
+ suppliedUsd: '0',
167
+ borrowedUsd: '0',
168
+ borrowLimitUsd: '0',
169
+ leftToBorrowUsd: '0',
170
+ ratio: '0',
171
+ minRatio: '0',
172
+ netApy: '0',
173
+ incentiveUsd: '0',
174
+ totalInterestUsd: '0',
175
+ isSubscribedToAutomation: false,
176
+ automationResubscribeRequired: false,
177
+ lastUpdated: Date.now(),
178
+ };
179
+
180
+ const parseUserData = (userPositionData: FluidView.UserPositionStructOutputStruct, vaultData: FluidMarketData): FluidVaultData => {
181
+ const {
182
+ assetsData,
183
+ marketData,
184
+ } = vaultData;
185
+
186
+ const payload = {
187
+ owner: userPositionData.owner,
188
+ vaultId: marketData.vaultId,
189
+ ...EMPTY_FLUID_DATA,
190
+ lastUpdated: Date.now(),
191
+ };
192
+ const collAsset = getAssetInfo(marketData.collAsset0);
193
+ const debtAsset = getAssetInfo(marketData.debtAsset0);
194
+
195
+ const supplied = getEthAmountForDecimals(userPositionData.supply, collAsset.decimals);
196
+ const borrowed = getEthAmountForDecimals(userPositionData.borrow, debtAsset.decimals);
197
+
198
+ const collUsedAsset: FluidUsedAsset = {
199
+ ...EMPTY_USED_ASSET,
200
+ symbol: collAsset.symbol,
201
+ collateral: true,
202
+ supplied,
203
+ suppliedUsd: new Dec(supplied).mul(assetsData[collAsset.symbol].price).toString(),
204
+ isSupplied: new Dec(supplied).gt(0),
205
+ };
206
+
207
+ const debtUsedAsset: FluidUsedAsset = {
208
+ ...EMPTY_USED_ASSET,
209
+ symbol: debtAsset.symbol,
210
+ collateral: false,
211
+ borrowed,
212
+ borrowedUsd: new Dec(borrowed).mul(assetsData[debtAsset.symbol].price).toString(),
213
+ isBorrowed: new Dec(borrowed).gt(0),
214
+ };
215
+
216
+ const usedAssets: FluidUsedAssets = {
217
+ [collAsset.symbol]: collUsedAsset,
218
+ [debtAsset.symbol]: debtUsedAsset,
219
+ };
220
+
221
+ return {
222
+ ...payload,
223
+ usedAssets,
224
+ ...(getFluidAggregatedData({
225
+ usedAssets,
226
+ assetsData,
227
+ marketData,
228
+ }) as FluidAggregatedVaultData),
229
+ };
230
+ };
231
+
232
+ export const getFluidMarketData = async (web3: Web3, network: NetworkNumber, market: FluidMarketInfo, mainnetWeb3: Web3) => {
233
+ const view = FluidViewContract(web3, network);
234
+
235
+ const data = await view.methods.getVaultData(market.marketAddress).call();
236
+
237
+ return parseMarketData(web3, data, network, mainnetWeb3);
238
+ };
239
+
240
+ export const getFluidVaultIdsForUser = async (web3: Web3,
241
+ network:NetworkNumber,
242
+ user: EthAddress): Promise<string[]> => {
243
+ const view = FluidViewContract(web3, network);
244
+
245
+ return view.methods.getUserNftIds(user).call();
246
+ };
247
+
248
+
249
+ export const getFluidPosition = async (
250
+ web3: Web3,
251
+ network: NetworkNumber,
252
+ vaultId: string,
253
+ extractedState: {
254
+ assetsData: FluidAssetsData
255
+ marketData: InnerFluidMarketData,
256
+ },
257
+ ): Promise<FluidVaultData> => {
258
+ const view = FluidViewContract(web3, network);
259
+
260
+ const data = await view.methods.getPositionByNftId(vaultId).call();
261
+
262
+ const userPositionData = data[0];
263
+
264
+ return parseUserData(userPositionData, extractedState);
265
+ };
266
+
267
+ export const getFluidPositionWithMarket = async (web3: Web3, network: NetworkNumber, vaultId: string, mainnetWeb3: Web3) => {
268
+ const view = FluidViewContract(web3, network);
269
+ const data = await view.methods.getPositionByNftId(vaultId).call();
270
+ const marketData = await parseMarketData(web3, data.vault, network, mainnetWeb3);
271
+ const userData = parseUserData(data.position, marketData);
272
+
273
+ return {
274
+ userData,
275
+ marketData,
276
+ };
277
+ };
278
+
279
+ export const getAllFluidMarketDataChunked = async (network: NetworkNumber, web3: Web3, mainnetWeb3: Web3) => {
280
+ const versions = getFluidVersionsDataForNetwork(network);
281
+ const view = FluidViewContract(web3, network);
282
+ const calls = versions.map((version) => ({
283
+ target: view.options.address,
284
+ abiItem: view.options.jsonInterface.find((item) => item.name === 'getVaultData'),
285
+ params: [version.marketAddress],
286
+ }));
287
+
288
+ const data = await chunkAndMulticall(calls, 10, 'latest', web3, network);
289
+ // @ts-ignore
290
+ return Promise.all(data.map(async (item, i) => parseMarketData(web3, item.vaultData, network, mainnetWeb3)));
291
+ };
292
+
293
+ export const getFluidTokenData = async (web3: Web3, network: NetworkNumber, token: string) => {
294
+ const view = FluidViewContract(web3, network);
295
+ const fTokenAddress = getFTokenAddress(token, network);
296
+ const data = await view.methods.getFTokenData(fTokenAddress).call();
297
+ const supplyRate = new Dec(data.supplyRate).div(100).toString();
298
+ const rewardsRate = new Dec(data.rewardsRate).div(1e12).toString();
299
+ const decimals = data.decimals;
300
+
301
+ const depositRate = new Dec(getEthAmountForDecimals(data.convertToShares, decimals)).toString();
302
+ const withdrawRate = new Dec(getEthAmountForDecimals(data.convertToAssets, decimals)).toString();
303
+
304
+ return {
305
+ fTokenAddress,
306
+ fTokenSymbol: data.symbol,
307
+ decimals,
308
+ totalDeposited: getEthAmountForDecimals(data.totalAssets, decimals),
309
+ withdrawable: getEthAmountForDecimals(data.withdrawable, decimals),
310
+ apy: new Dec(supplyRate).add(rewardsRate).toString(),
311
+ depositRate,
312
+ withdrawRate,
313
+ };
314
+ };
315
+
316
+ export const getFluidDepositData = async (web3: Web3, network: NetworkNumber, token: string, address: EthAddress) => {
317
+ const view = FluidViewContract(web3, network);
318
+ const fTokenAddress = getFTokenAddress(token, network);
319
+ const { fTokenData, userPosition } = await view.methods.getUserEarnPositionWithFToken(fTokenAddress, address).call();
320
+
321
+ const supplyRate = new Dec(fTokenData.supplyRate).div(100).toString();
322
+ const rewardsRate = new Dec(fTokenData.rewardsRate).div(1e12).toString();
323
+ const decimals = fTokenData.decimals;
324
+
325
+ const depositRate = new Dec(getEthAmountForDecimals(fTokenData.convertToShares, decimals)).toString();
326
+ const withdrawRate = new Dec(getEthAmountForDecimals(fTokenData.convertToAssets, decimals)).toString();
327
+
328
+ return {
329
+ fTokenAddress,
330
+ fTokenSymbol: fTokenData.symbol,
331
+ decimals,
332
+ totalDeposited: getEthAmountForDecimals(fTokenData.totalAssets, decimals),
333
+ withdrawable: getEthAmountForDecimals(fTokenData.withdrawable, decimals),
334
+ apy: new Dec(supplyRate).add(rewardsRate).toString(),
335
+ depositRate,
336
+ withdrawRate,
337
+ deposited: getEthAmountForDecimals(userPosition.underlyingAssets, decimals),
338
+ depositedShares: getEthAmountForDecimals(userPosition.fTokenShares, decimals),
339
+ };
340
+ };
341
+
342
+ export const getUserPositions = async (web3: Web3, network: NetworkNumber, user: EthAddress, mainnetWeb3: Web3) => {
343
+ const view = FluidViewContract(web3, network);
344
+
345
+ const data = await view.methods.getUserPositions(user).call();
346
+
347
+ const parsedMarketData = await Promise.all(data.vaults.map(async (vaultData) => parseMarketData(web3, vaultData, network, mainnetWeb3)));
348
+
349
+ const userData = data.positions.map((position, i) => ({ ...parseUserData(position, parsedMarketData[i]), nftId: position.nftId }));
350
+
351
+ return parsedMarketData.map((market, i) => ({
352
+ marketData: market,
353
+ userData: userData[i],
354
+ }));
355
355
  };