@defisaver/positions-sdk 0.0.201-fluid-dev-4 → 0.0.201-fluid-dev-6

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