@defisaver/positions-sdk 2.0.12 → 2.0.14-dev-portfolio

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 (170) hide show
  1. package/.mocharc.json +4 -4
  2. package/.nvmrc +1 -1
  3. package/CLAUDE.md +32 -0
  4. package/README.md +64 -64
  5. package/cjs/aaveV2/index.js +1 -0
  6. package/cjs/aaveV3/index.d.ts +12 -0
  7. package/cjs/aaveV3/index.js +93 -1
  8. package/cjs/claiming/aaveV3.d.ts +9 -0
  9. package/cjs/claiming/aaveV3.js +148 -0
  10. package/cjs/claiming/compV3.d.ts +15 -0
  11. package/cjs/claiming/compV3.js +34 -0
  12. package/cjs/claiming/index.d.ts +6 -0
  13. package/cjs/claiming/index.js +46 -0
  14. package/cjs/claiming/king.d.ts +4 -0
  15. package/cjs/claiming/king.js +72 -0
  16. package/cjs/claiming/morphoBlue.d.ts +6 -0
  17. package/cjs/claiming/morphoBlue.js +113 -0
  18. package/cjs/claiming/spark.d.ts +6 -0
  19. package/cjs/claiming/spark.js +188 -0
  20. package/cjs/config/contracts.d.ts +2667 -0
  21. package/cjs/config/contracts.js +103 -2
  22. package/cjs/constants/index.d.ts +4 -0
  23. package/cjs/constants/index.js +6 -2
  24. package/cjs/contracts.d.ts +2882 -23
  25. package/cjs/contracts.js +10 -1
  26. package/cjs/helpers/morphoBlueHelpers/index.js +66 -66
  27. package/cjs/index.d.ts +2 -1
  28. package/cjs/index.js +3 -1
  29. package/cjs/liquity/index.d.ts +11 -0
  30. package/cjs/liquity/index.js +39 -1
  31. package/cjs/liquityV2/index.d.ts +8 -0
  32. package/cjs/liquityV2/index.js +161 -0
  33. package/cjs/markets/aave/marketAssets.js +1 -1
  34. package/cjs/markets/compound/marketsAssets.js +2 -2
  35. package/cjs/markets/spark/marketAssets.js +1 -1
  36. package/cjs/morphoBlue/index.d.ts +5 -0
  37. package/cjs/morphoBlue/index.js +38 -4
  38. package/cjs/portfolio/index.d.ts +6 -1
  39. package/cjs/portfolio/index.js +256 -10
  40. package/cjs/services/utils.d.ts +5 -0
  41. package/cjs/services/utils.js +33 -1
  42. package/cjs/services/viem.d.ts +12 -12
  43. package/cjs/spark/index.js +1 -0
  44. package/cjs/staking/staking.js +3 -1
  45. package/cjs/types/claiming.d.ts +93 -0
  46. package/cjs/types/claiming.js +27 -0
  47. package/cjs/umbrella/index.d.ts +5 -0
  48. package/cjs/umbrella/index.js +50 -0
  49. package/cjs/umbrella/umbrellaUtils.d.ts +22 -0
  50. package/cjs/umbrella/umbrellaUtils.js +34 -0
  51. package/esm/aaveV2/index.js +1 -0
  52. package/esm/aaveV3/index.d.ts +12 -0
  53. package/esm/aaveV3/index.js +91 -1
  54. package/esm/claiming/aaveV3.d.ts +9 -0
  55. package/esm/claiming/aaveV3.js +139 -0
  56. package/esm/claiming/compV3.d.ts +15 -0
  57. package/esm/claiming/compV3.js +30 -0
  58. package/esm/claiming/index.d.ts +6 -0
  59. package/esm/claiming/index.js +6 -0
  60. package/esm/claiming/king.d.ts +4 -0
  61. package/esm/claiming/king.js +64 -0
  62. package/esm/claiming/morphoBlue.d.ts +6 -0
  63. package/esm/claiming/morphoBlue.js +104 -0
  64. package/esm/claiming/spark.d.ts +6 -0
  65. package/esm/claiming/spark.js +179 -0
  66. package/esm/config/contracts.d.ts +2667 -0
  67. package/esm/config/contracts.js +102 -1
  68. package/esm/constants/index.d.ts +4 -0
  69. package/esm/constants/index.js +5 -1
  70. package/esm/contracts.d.ts +2882 -23
  71. package/esm/contracts.js +9 -0
  72. package/esm/helpers/morphoBlueHelpers/index.js +66 -66
  73. package/esm/index.d.ts +2 -1
  74. package/esm/index.js +2 -1
  75. package/esm/liquity/index.d.ts +11 -0
  76. package/esm/liquity/index.js +38 -1
  77. package/esm/liquityV2/index.d.ts +8 -0
  78. package/esm/liquityV2/index.js +162 -1
  79. package/esm/markets/aave/marketAssets.js +1 -1
  80. package/esm/markets/compound/marketsAssets.js +2 -2
  81. package/esm/markets/spark/marketAssets.js +1 -1
  82. package/esm/morphoBlue/index.d.ts +5 -0
  83. package/esm/morphoBlue/index.js +38 -5
  84. package/esm/portfolio/index.d.ts +6 -1
  85. package/esm/portfolio/index.js +260 -14
  86. package/esm/services/utils.d.ts +5 -0
  87. package/esm/services/utils.js +31 -0
  88. package/esm/services/viem.d.ts +12 -12
  89. package/esm/spark/index.js +1 -0
  90. package/esm/staking/staking.js +3 -1
  91. package/esm/types/claiming.d.ts +93 -0
  92. package/esm/types/claiming.js +24 -0
  93. package/esm/umbrella/index.d.ts +5 -0
  94. package/esm/umbrella/index.js +46 -0
  95. package/esm/umbrella/umbrellaUtils.d.ts +22 -0
  96. package/esm/umbrella/umbrellaUtils.js +28 -0
  97. package/package.json +47 -47
  98. package/src/aaveV2/index.ts +237 -236
  99. package/src/aaveV3/index.ts +586 -488
  100. package/src/claiming/aaveV3.ts +163 -0
  101. package/src/claiming/compV3.ts +23 -0
  102. package/src/claiming/index.ts +13 -0
  103. package/src/claiming/king.ts +66 -0
  104. package/src/claiming/morphoBlue.ts +119 -0
  105. package/src/claiming/spark.ts +226 -0
  106. package/src/compoundV2/index.ts +240 -240
  107. package/src/compoundV3/index.ts +270 -270
  108. package/src/config/contracts.ts +1208 -1108
  109. package/src/constants/index.ts +10 -6
  110. package/src/contracts.ts +121 -108
  111. package/src/curveUsd/index.ts +250 -250
  112. package/src/eulerV2/index.ts +314 -314
  113. package/src/exchange/index.ts +25 -25
  114. package/src/fluid/index.ts +1568 -1568
  115. package/src/helpers/aaveHelpers/index.ts +170 -170
  116. package/src/helpers/compoundHelpers/index.ts +261 -261
  117. package/src/helpers/curveUsdHelpers/index.ts +40 -40
  118. package/src/helpers/eulerHelpers/index.ts +259 -259
  119. package/src/helpers/fluidHelpers/index.ts +324 -324
  120. package/src/helpers/index.ts +10 -10
  121. package/src/helpers/liquityV2Helpers/index.ts +80 -80
  122. package/src/helpers/llamaLendHelpers/index.ts +53 -53
  123. package/src/helpers/makerHelpers/index.ts +52 -52
  124. package/src/helpers/morphoBlueHelpers/index.ts +390 -390
  125. package/src/helpers/sparkHelpers/index.ts +155 -155
  126. package/src/index.ts +47 -45
  127. package/src/liquity/index.ts +160 -105
  128. package/src/liquityV2/index.ts +583 -408
  129. package/src/llamaLend/index.ts +296 -296
  130. package/src/maker/index.ts +223 -223
  131. package/src/markets/aave/index.ts +116 -116
  132. package/src/markets/aave/marketAssets.ts +49 -49
  133. package/src/markets/compound/index.ts +227 -227
  134. package/src/markets/compound/marketsAssets.ts +90 -90
  135. package/src/markets/curveUsd/index.ts +69 -69
  136. package/src/markets/euler/index.ts +26 -26
  137. package/src/markets/fluid/index.ts +2456 -2456
  138. package/src/markets/index.ts +25 -25
  139. package/src/markets/liquityV2/index.ts +102 -102
  140. package/src/markets/llamaLend/contractAddresses.ts +141 -141
  141. package/src/markets/llamaLend/index.ts +235 -235
  142. package/src/markets/morphoBlue/index.ts +895 -895
  143. package/src/markets/spark/index.ts +29 -29
  144. package/src/markets/spark/marketAssets.ts +11 -11
  145. package/src/moneymarket/moneymarketCommonService.ts +80 -80
  146. package/src/morphoBlue/index.ts +259 -222
  147. package/src/portfolio/index.ts +533 -285
  148. package/src/services/priceService.ts +159 -159
  149. package/src/services/utils.ts +100 -64
  150. package/src/services/viem.ts +32 -32
  151. package/src/setup.ts +8 -8
  152. package/src/spark/index.ts +457 -456
  153. package/src/staking/staking.ts +194 -193
  154. package/src/types/aave.ts +194 -194
  155. package/src/types/claiming.ts +109 -0
  156. package/src/types/common.ts +88 -88
  157. package/src/types/compound.ts +136 -136
  158. package/src/types/curveUsd.ts +121 -121
  159. package/src/types/euler.ts +174 -174
  160. package/src/types/fluid.ts +450 -450
  161. package/src/types/index.ts +11 -11
  162. package/src/types/liquity.ts +30 -30
  163. package/src/types/liquityV2.ts +126 -126
  164. package/src/types/llamaLend.ts +157 -157
  165. package/src/types/maker.ts +63 -63
  166. package/src/types/morphoBlue.ts +194 -194
  167. package/src/types/portfolio.ts +60 -60
  168. package/src/types/spark.ts +137 -137
  169. package/src/umbrella/index.ts +70 -0
  170. package/src/umbrella/umbrellaUtils.ts +30 -0
@@ -1,488 +1,586 @@
1
- import { assetAmountInEth, assetAmountInWei, getAssetInfo } from '@defisaver/tokens';
2
- import { Client } from 'viem';
3
- import Dec from 'decimal.js';
4
- import {
5
- AaveIncentiveDataProviderV3ContractViem,
6
- AaveV3ViewContractViem,
7
- createViemContractFromConfigFunc,
8
- } from '../contracts';
9
- import { aaveAnyGetAggregatedPositionData, aaveV3IsInIsolationMode, aaveV3IsInSiloedMode } from '../helpers/aaveHelpers';
10
- import { AAVE_V3 } from '../markets/aave';
11
- import { aprToApy, calculateBorrowingAssetLimit } from '../moneymarket';
12
- import {
13
- ethToWeth, isEnabledOnBitmap, isLayer2Network, wethToEth, wethToEthByAddress,
14
- } from '../services/utils';
15
- import { getStakingApy, STAKING_ASSETS } from '../staking';
16
- import {
17
- AaveMarketInfo,
18
- AaveV3AssetData,
19
- AaveV3AssetsData,
20
- AaveV3MarketData,
21
- AaveV3PositionData,
22
- AaveV3UsedAsset,
23
- AaveV3UsedAssets,
24
- EModeCategoriesData,
25
- EModeCategoryData,
26
- EModeCategoryDataMapping,
27
- } from '../types/aave';
28
- import {
29
- Blockish, EthAddress, EthereumProvider, NetworkNumber, PositionBalances,
30
- } from '../types/common';
31
- import { getViemProvider, setViemBlockNumber } from '../services/viem';
32
-
33
- export const aaveV3EmodeCategoriesMapping = (extractedState: any, usedAssets: AaveV3UsedAssets) => {
34
- const { eModeCategoriesData }: { assetsData: AaveV3AssetsData, eModeCategoriesData: EModeCategoriesData } = extractedState;
35
- const usedAssetsValues = Object.values(usedAssets);
36
-
37
- const categoriesMapping: { [key: number]: EModeCategoryDataMapping } = {};
38
-
39
- Object.values(eModeCategoriesData).forEach((e: EModeCategoryData) => {
40
- const borrowingOnlyFromCategory = e.id === 0
41
- ? true
42
- : !usedAssetsValues.filter(u => u.isBorrowed && !e.borrowAssets.includes(u.symbol)).length;
43
- const afterEnteringCategory = aaveAnyGetAggregatedPositionData({
44
- ...extractedState, usedAssets, eModeCategory: e.id,
45
- });
46
- const willStayOverCollateralized = new Dec(afterEnteringCategory.ratio).eq(0) || new Dec(afterEnteringCategory.ratio).gt(afterEnteringCategory.liqPercent);
47
- const enteringTerms = [borrowingOnlyFromCategory, willStayOverCollateralized];
48
- categoriesMapping[e.id] = {
49
- enteringTerms,
50
- canEnterCategory: !enteringTerms.includes(false),
51
- id: e.id,
52
- enabledData: {
53
- ratio: afterEnteringCategory.ratio,
54
- liqRatio: afterEnteringCategory.liqRatio,
55
- liqPercent: afterEnteringCategory.liqPercent,
56
- collRatio: afterEnteringCategory.collRatio,
57
- },
58
- };
59
- });
60
- return categoriesMapping;
61
- };
62
-
63
- export async function _getAaveV3MarketData(provider: Client, network: NetworkNumber, market: AaveMarketInfo, blockNumber: 'latest' | number = 'latest'): Promise<AaveV3MarketData> {
64
- const _addresses = market.assets.map(a => getAssetInfo(ethToWeth(a), network).address);
65
-
66
- const isL2 = isLayer2Network(network);
67
- const loanInfoContract = AaveV3ViewContractViem(provider, network);
68
- const aaveIncentivesContract = AaveIncentiveDataProviderV3ContractViem(provider, network);
69
- const marketAddress = market.providerAddress;
70
- const networksWithIncentives = [NetworkNumber.Eth, NetworkNumber.Arb, NetworkNumber.Opt, NetworkNumber.Linea];
71
-
72
- // eslint-disable-next-line prefer-const
73
- let [loanInfo, eModesInfo, isBorrowAllowed, rewardInfo] = await Promise.all([
74
- loanInfoContract.read.getFullTokensInfo([marketAddress, _addresses as EthAddress[]], setViemBlockNumber(blockNumber)),
75
- loanInfoContract.read.getAllEmodes([marketAddress], setViemBlockNumber(blockNumber)),
76
- loanInfoContract.read.isBorrowAllowed([marketAddress], setViemBlockNumber(blockNumber)), // Used on L2s check for PriceOracleSentinel (mainnet will always return true)
77
- networksWithIncentives.includes(network) ? aaveIncentivesContract.read.getReservesIncentivesData([marketAddress], setViemBlockNumber(blockNumber)) : null,
78
- ]);
79
- isBorrowAllowed = isLayer2Network(network) ? isBorrowAllowed : true;
80
-
81
- const eModeCategoriesData: EModeCategoriesData = {};
82
- for (let i = 0; i < eModesInfo.length; i++) {
83
- if (!eModesInfo[i].label) break;
84
- eModeCategoriesData[i + 1] = {
85
- label: eModesInfo[i].label,
86
- id: i + 1,
87
- liquidationBonus: new Dec(eModesInfo[i].liquidationBonus).div(10000).toString(),
88
- liquidationRatio: new Dec(eModesInfo[i].liquidationThreshold).div(10000).toString(),
89
- collateralFactor: new Dec(eModesInfo[i].ltv).div(10000).toString(),
90
- borrowableBitmap: eModesInfo[i].borrowableBitmap.toString(),
91
- collateralBitmap: eModesInfo[i].collateralBitmap.toString(),
92
- borrowAssets: [],
93
- collateralAssets: [],
94
- };
95
- }
96
-
97
- if (networksWithIncentives.includes(network) && rewardInfo) {
98
- rewardInfo = rewardInfo.reduce((all: any, _market: any) => {
99
- // eslint-disable-next-line no-param-reassign
100
- all[_market.underlyingAsset] = _market;
101
- return all;
102
- }, {});
103
- }
104
-
105
- const assetsData: AaveV3AssetData[] = await Promise.all(loanInfo
106
- .map(async (tokenMarket, i) => {
107
- const symbol = market.assets[i];
108
-
109
- // eslint-disable-next-line guard-for-in
110
- for (const eModeIndex in eModeCategoriesData) {
111
- if (isEnabledOnBitmap(Number(eModeCategoriesData[eModeIndex].collateralBitmap), Number(tokenMarket.assetId))) eModeCategoriesData[eModeIndex].collateralAssets.push(symbol);
112
- if (isEnabledOnBitmap(Number(eModeCategoriesData[eModeIndex].borrowableBitmap), Number(tokenMarket.assetId))) eModeCategoriesData[eModeIndex].borrowAssets.push(symbol);
113
- }
114
-
115
- const borrowCap = tokenMarket.borrowCap.toString();
116
-
117
- const borrowCapInWei = new Dec(assetAmountInWei(borrowCap, symbol));
118
- let marketLiquidity = borrowCapInWei.lt(new Dec(tokenMarket.totalSupply.toString()))
119
- ? assetAmountInEth(borrowCapInWei
120
- .sub(tokenMarket.totalBorrow.toString())
121
- .toString(), symbol)
122
- : assetAmountInEth(new Dec(tokenMarket.totalSupply.toString())
123
- .sub(tokenMarket.totalBorrow.toString())
124
- .toString(), symbol);
125
-
126
- if (new Dec(marketLiquidity).lt(0)) {
127
- marketLiquidity = '0';
128
- }
129
- return ({
130
- symbol,
131
- isIsolated: new Dec(tokenMarket.debtCeilingForIsolationMode.toString()).gt(0),
132
- debtCeilingForIsolationMode: new Dec(tokenMarket.debtCeilingForIsolationMode.toString()).div(100).toString(),
133
- isSiloed: tokenMarket.isSiloedForBorrowing,
134
- isolationModeTotalDebt: new Dec(tokenMarket.isolationModeTotalDebt.toString()).div(100).toString(),
135
- assetId: Number(tokenMarket.assetId),
136
- underlyingTokenAddress: tokenMarket.underlyingTokenAddress,
137
- supplyRate: aprToApy(new Dec(tokenMarket.supplyRate.toString()).div(1e25).toString()),
138
- borrowRate: aprToApy(new Dec(tokenMarket.borrowRateVariable.toString()).div(1e25).toString()),
139
- borrowRateStable: aprToApy(new Dec(tokenMarket.borrowRateStable.toString()).div(1e25).toString()),
140
- collateralFactor: new Dec(tokenMarket.collateralFactor.toString()).div(10000).toString(),
141
- liquidationBonus: new Dec(tokenMarket.liquidationBonus.toString()).div(10000).toString(),
142
- liquidationRatio: new Dec(tokenMarket.liquidationRatio.toString()).div(10000).toString(),
143
- marketLiquidity,
144
- utilization: new Dec(tokenMarket.totalBorrow.toString()).times(100).div(new Dec(tokenMarket.totalSupply.toString())).toString(),
145
- usageAsCollateralEnabled: tokenMarket.usageAsCollateralEnabled,
146
- supplyCap: tokenMarket.supplyCap.toString(),
147
- borrowCap,
148
- totalSupply: assetAmountInEth(tokenMarket.totalSupply.toString(), symbol),
149
- isInactive: !tokenMarket.isActive,
150
- isFrozen: tokenMarket.isFrozen,
151
- isPaused: tokenMarket.isPaused,
152
- canBeBorrowed: tokenMarket.isActive && !tokenMarket.isPaused && !tokenMarket.isFrozen && tokenMarket.borrowingEnabled && isBorrowAllowed,
153
- canBeSupplied: tokenMarket.isActive && !tokenMarket.isPaused && !tokenMarket.isFrozen,
154
- canBeWithdrawn: tokenMarket.isActive && !tokenMarket.isPaused,
155
- canBePayBacked: tokenMarket.isActive && !tokenMarket.isPaused,
156
- disabledStableBorrowing: !tokenMarket.stableBorrowRateEnabled,
157
- totalBorrow: assetAmountInEth(tokenMarket.totalBorrow.toString(), symbol),
158
- totalBorrowVar: assetAmountInEth(tokenMarket.totalBorrowVar.toString(), symbol),
159
- price: new Dec(tokenMarket.price.toString()).div(1e8).toString(), // is actually price in USD
160
- isolationModeBorrowingEnabled: tokenMarket.isolationModeBorrowingEnabled,
161
- isFlashLoanEnabled: tokenMarket.isFlashLoanEnabled,
162
- aTokenAddress: tokenMarket.aTokenAddress,
163
- });
164
- }));
165
-
166
-
167
- // Get incentives data
168
- await Promise.all(assetsData.map(async (_market: AaveV3AssetData) => {
169
- /* eslint-disable no-param-reassign */
170
- // @ts-ignore
171
- const rewardForMarket = rewardInfo?.[_market.underlyingTokenAddress];
172
- const isStakingAsset = STAKING_ASSETS.includes(_market.symbol);
173
- if (isStakingAsset) {
174
- _market.incentiveSupplyApy = await getStakingApy(_market.symbol);
175
- _market.incentiveSupplyToken = _market.symbol;
176
- if (!_market.supplyIncentives) {
177
- _market.supplyIncentives = [];
178
- }
179
- _market.supplyIncentives.push({
180
- apy: _market.incentiveSupplyApy || '0',
181
- token: _market.symbol,
182
- incentiveKind: 'staking',
183
- });
184
- }
185
-
186
- if (_market.canBeBorrowed && _market.incentiveSupplyApy) {
187
- _market.incentiveBorrowApy = _market.incentiveSupplyApy;
188
- _market.incentiveBorrowToken = _market.incentiveSupplyToken;
189
- if (!_market.borrowIncentives) {
190
- _market.borrowIncentives = [];
191
- }
192
- _market.borrowIncentives.push({
193
- apy: _market.incentiveBorrowApy,
194
- token: _market.incentiveBorrowToken!!,
195
- incentiveKind: 'reward',
196
- });
197
- }
198
-
199
- if (_market.symbol === 'USDe') {
200
- const merklApy = await getStakingApy(_market.symbol);
201
- if (!_market.supplyIncentives) {
202
- _market.supplyIncentives = [];
203
- }
204
- _market.supplyIncentives.push({
205
- apy: merklApy || '0',
206
- token: _market.symbol,
207
- incentiveKind: 'reward',
208
- });
209
- }
210
-
211
- if (!rewardForMarket) return;
212
- // @ts-ignore
213
- rewardForMarket.aIncentiveData.rewardsTokenInformation.forEach(supplyRewardData => {
214
- if (supplyRewardData) {
215
- if (+(supplyRewardData.emissionEndTimestamp.toString()) * 1000 < Date.now()) return;
216
- _market.incentiveSupplyToken = supplyRewardData.rewardTokenSymbol;
217
- // reward token is aave asset
218
- if (supplyRewardData.rewardTokenSymbol.startsWith('a') && supplyRewardData.rewardTokenSymbol.includes(_market.symbol)) _market.incentiveSupplyToken = _market.symbol;
219
- const supplyEmissionPerSecond = supplyRewardData.emissionPerSecond;
220
- const supplyRewardPrice = new Dec(supplyRewardData.rewardPriceFeed).div(10 ** +supplyRewardData.priceFeedDecimals)
221
- .toString();
222
- const rewardApy = new Dec(supplyEmissionPerSecond).div((10 ** +supplyRewardData.rewardTokenDecimals) / 100)
223
- .mul(365 * 24 * 3600)
224
- .mul(supplyRewardPrice)
225
- .div(_market.price)
226
- .div(_market.totalSupply)
227
- .toString();
228
- _market.incentiveSupplyApy = new Dec(_market.incentiveSupplyApy || '0').add(rewardApy)
229
- .toString();
230
-
231
- if (!_market.supplyIncentives) {
232
- _market.supplyIncentives = [];
233
- }
234
- _market.supplyIncentives.push({
235
- token: supplyRewardData.rewardTokenSymbol,
236
- apy: rewardApy,
237
- incentiveKind: 'reward',
238
- });
239
- }
240
- });
241
- // @ts-ignore
242
- rewardForMarket.vIncentiveData.rewardsTokenInformation.forEach(borrowRewardData => {
243
- if (borrowRewardData) {
244
- if (+(borrowRewardData.emissionEndTimestamp.toString()) * 1000 < Date.now()) return;
245
- _market.incentiveBorrowToken = borrowRewardData.rewardTokenSymbol;
246
- if (borrowRewardData.rewardTokenSymbol.startsWith('a') && borrowRewardData.rewardTokenSymbol.includes(_market.symbol)) _market.incentiveBorrowToken = _market.symbol;
247
- const supplyEmissionPerSecond = borrowRewardData.emissionPerSecond;
248
- const supplyRewardPrice = new Dec(borrowRewardData.rewardPriceFeed).div(10 ** +borrowRewardData.priceFeedDecimals)
249
- .toString();
250
- const rewardApy = new Dec(supplyEmissionPerSecond).div((10 ** +borrowRewardData.rewardTokenDecimals) / 100)
251
- .mul(365 * 24 * 3600)
252
- .mul(supplyRewardPrice)
253
- .div(_market.price)
254
- .div(_market.totalBorrowVar)
255
- .toString();
256
- _market.incentiveBorrowApy = new Dec(_market.incentiveBorrowApy || '0').add(rewardApy)
257
- .toString();
258
-
259
- if (!_market.borrowIncentives) {
260
- _market.borrowIncentives = [];
261
- }
262
- _market.borrowIncentives.push({
263
- token: borrowRewardData.rewardTokenSymbol,
264
- apy: rewardApy,
265
- incentiveKind: 'reward',
266
- });
267
- }
268
- });
269
- }));
270
-
271
- const payload: AaveV3AssetsData = {};
272
- // Sort by market size
273
- assetsData
274
- .sort((a, b) => {
275
- const aMarket = new Dec(a.price).times(a.totalSupply).toString();
276
- const bMarket = new Dec(b.price).times(b.totalSupply).toString();
277
-
278
- return new Dec(bMarket).minus(aMarket).toNumber();
279
- })
280
- .forEach((assetData: AaveV3AssetData, i) => {
281
- payload[assetData.symbol] = { ...assetData, sortIndex: i };
282
- });
283
-
284
- eModeCategoriesData[0] = {
285
- id: 0,
286
- label: '',
287
- liquidationBonus: '0',
288
- liquidationRatio: '0',
289
- collateralFactor: '0',
290
- collateralAssets: assetsData.map((a) => a.symbol),
291
- borrowAssets: assetsData.map((a) => a.symbol),
292
- };
293
-
294
- return { assetsData: payload, eModeCategoriesData };
295
- }
296
-
297
- export async function getAaveV3MarketData(provider: EthereumProvider, network: NetworkNumber, market: AaveMarketInfo, blockNumber: 'latest' | number = 'latest'): Promise<AaveV3MarketData> {
298
- return _getAaveV3MarketData(getViemProvider(provider, network), network, market, blockNumber);
299
- }
300
-
301
- export const EMPTY_AAVE_DATA = {
302
- usedAssets: {},
303
- suppliedUsd: '0',
304
- borrowedUsd: '0',
305
- borrowLimitUsd: '0',
306
- leftToBorrowUsd: '0',
307
- ratio: '0',
308
- minRatio: '0',
309
- netApy: '0',
310
- incentiveUsd: '0',
311
- totalInterestUsd: '0',
312
- isSubscribedToAutomation: false,
313
- automationResubscribeRequired: false,
314
- eModeCategory: 0,
315
- isInIsolationMode: false,
316
- isInSiloedMode: false,
317
- totalSupplied: '0',
318
- eModeCategories: [],
319
- collRatio: '0',
320
- suppliedCollateralUsd: '0',
321
- };
322
-
323
- export const _getAaveV3AccountBalances = async (provider: Client, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress): Promise<PositionBalances> => {
324
- let balances: PositionBalances = {
325
- collateral: {},
326
- debt: {},
327
- };
328
-
329
- if (!address) {
330
- return balances;
331
- }
332
-
333
- const loanInfoContract = AaveV3ViewContractViem(provider, network, block);
334
-
335
- const market = AAVE_V3(network);
336
- const marketAddress = market.providerAddress;
337
- // @ts-ignore
338
- const protocolDataProviderContract = createViemContractFromConfigFunc(market.protocolData, market.protocolDataAddress)(provider, network);
339
-
340
- const reserveTokens = await protocolDataProviderContract.read.getAllReservesTokens(setViemBlockNumber(block));
341
- const symbols = reserveTokens.map(({ symbol }: { symbol: string }) => symbol);
342
- const _addresses = reserveTokens.map(({ tokenAddress }: { tokenAddress: EthAddress }) => tokenAddress);
343
-
344
- // split addresses in half to avoid gas limit by multicall
345
- const middleAddressIndex = Math.floor(_addresses.length / 2);
346
-
347
- const [tokenBalances1, tokenBalances2] = await Promise.all([
348
- loanInfoContract.read.getTokenBalances([marketAddress, address, _addresses.slice(0, middleAddressIndex)], setViemBlockNumber(block)),
349
- loanInfoContract.read.getTokenBalances([marketAddress, address, _addresses.slice(middleAddressIndex, _addresses.length)], setViemBlockNumber(block)),
350
- ]);
351
-
352
- const loanInfo = [...tokenBalances1, ...tokenBalances2];
353
-
354
- loanInfo.forEach((tokenInfo: any, i: number) => {
355
- const asset = wethToEth(symbols[i]);
356
- const assetAddr = wethToEthByAddress(_addresses[i], network).toLowerCase();
357
-
358
- balances = {
359
- collateral: {
360
- ...balances.collateral,
361
- [addressMapping ? assetAddr : asset]: tokenInfo.balance.toString(),
362
- },
363
- debt: {
364
- ...balances.debt,
365
- [addressMapping ? assetAddr : asset]: new Dec(tokenInfo.borrowsStable.toString()).add(tokenInfo.borrowsVariable.toString()).toString(),
366
- },
367
- };
368
- });
369
-
370
- return balances;
371
- };
372
-
373
- export const getAaveV3AccountBalances = async (provider: EthereumProvider, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress): Promise<PositionBalances> => _getAaveV3AccountBalances(getViemProvider(provider, network), network, block, addressMapping, address);
374
-
375
- export const _getAaveV3AccountData = async (provider: Client, network: NetworkNumber, address: EthAddress, extractedState: any, blockNumber: 'latest' | number = 'latest'): Promise<AaveV3PositionData> => {
376
- const {
377
- selectedMarket: market, assetsData,
378
- } = extractedState;
379
- let payload: AaveV3PositionData = {
380
- ...EMPTY_AAVE_DATA,
381
- lastUpdated: Date.now(),
382
- };
383
- if (!address) {
384
- return payload;
385
- }
386
-
387
- const loanInfoContract = AaveV3ViewContractViem(provider, network);
388
- const lendingPoolContract = createViemContractFromConfigFunc(market.lendingPool, market.lendingPoolAddress)(provider, network);
389
- const marketAddress = market.providerAddress;
390
- const _addresses = market.assets.map((a: string[]) => getAssetInfo(ethToWeth(a), network).address);
391
-
392
- const middleAddressIndex = Math.floor(_addresses.length / 2); // split addresses in half to avoid gas limit by multicall
393
-
394
- const [eModeCategory, tokenBalances1, tokenBalances2] = await Promise.all([
395
- lendingPoolContract.read.getUserEMode([address], setViemBlockNumber(blockNumber)),
396
- loanInfoContract.read.getTokenBalances([marketAddress, address, _addresses.slice(0, middleAddressIndex) as EthAddress[]], setViemBlockNumber(blockNumber)),
397
- loanInfoContract.read.getTokenBalances([marketAddress, address, _addresses.slice(middleAddressIndex, _addresses.length) as EthAddress[]], setViemBlockNumber(blockNumber)),
398
- ]);
399
-
400
- const loanInfo = [...tokenBalances1, ...tokenBalances2];
401
-
402
- const usedAssets = {} as AaveV3UsedAssets;
403
- loanInfo.map(async (tokenInfo, i) => {
404
- const asset = market.assets[i];
405
- const isSupplied = tokenInfo.balance.toString() !== '0';
406
- const isBorrowed = tokenInfo.borrowsStable.toString() !== '0' || tokenInfo.borrowsVariable.toString() !== '0';
407
- if (!isSupplied && !isBorrowed) return;
408
-
409
- const supplied = assetAmountInEth(tokenInfo.balance.toString(), asset);
410
- const borrowedStable = assetAmountInEth(tokenInfo.borrowsStable.toString(), asset);
411
- const borrowedVariable = assetAmountInEth(tokenInfo.borrowsVariable.toString(), asset);
412
- const enabledAsCollateral = assetsData[asset].usageAsCollateralEnabled ? tokenInfo.enabledAsCollateral : false;
413
-
414
- let interestMode;
415
- if (borrowedVariable === '0' && borrowedStable !== '0') {
416
- interestMode = '1';
417
- } else if (borrowedVariable !== '0' && borrowedStable === '0') {
418
- interestMode = '2';
419
- } else {
420
- interestMode = 'both';
421
- }
422
- if (!usedAssets[asset]) usedAssets[asset] = {} as AaveV3UsedAsset;
423
- const borrowed = new Dec(borrowedStable).add(borrowedVariable).toString();
424
-
425
- usedAssets[asset] = {
426
- ...usedAssets[asset],
427
- symbol: asset,
428
- supplied,
429
- suppliedUsd: new Dec(supplied).mul(assetsData[asset].price).toString(),
430
- isSupplied,
431
- collateral: enabledAsCollateral,
432
- stableBorrowRate: aprToApy(new Dec(tokenInfo.stableBorrowRate.toString()).div(1e25).toString()),
433
- borrowedStable,
434
- borrowedVariable,
435
- borrowedUsdStable: new Dec(borrowedStable).mul(assetsData[asset].price).toString(),
436
- borrowedUsdVariable: new Dec(borrowedVariable).mul(assetsData[asset].price).toString(),
437
- borrowed,
438
- borrowedUsd: new Dec(new Dec(borrowedVariable).add(borrowedStable)).mul(assetsData[asset].price).toString(),
439
- isBorrowed,
440
- interestMode,
441
- };
442
- });
443
-
444
- payload.eModeCategory = +(eModeCategory as BigInt).toString();
445
- payload = {
446
- ...payload,
447
- usedAssets,
448
- ...aaveAnyGetAggregatedPositionData({
449
- ...extractedState, usedAssets, eModeCategory: payload.eModeCategory,
450
- }),
451
- };
452
- payload.eModeCategories = aaveV3EmodeCategoriesMapping(extractedState, usedAssets);
453
- payload.isInIsolationMode = aaveV3IsInIsolationMode({ usedAssets, assetsData });
454
- payload.isInSiloedMode = aaveV3IsInSiloedMode({ usedAssets, assetsData });
455
-
456
- payload.ratio = payload.borrowedUsd && payload.borrowedUsd !== '0'
457
- ? new Dec(payload.borrowLimitUsd).div(payload.borrowedUsd).mul(100).toString()
458
- : '0';
459
- payload.minRatio = '100';
460
- payload.collRatio = payload.borrowedUsd && payload.borrowedUsd !== '0'
461
- ? new Dec(payload.suppliedCollateralUsd).div(payload.borrowedUsd).mul(100).toString()
462
- : '0';
463
-
464
- // Calculate borrow limits per asset
465
- Object.values(payload.usedAssets).forEach((item) => {
466
- if (item.isBorrowed) {
467
- // eslint-disable-next-line no-param-reassign
468
- item.stableLimit = calculateBorrowingAssetLimit(item.borrowedUsdStable, payload.borrowLimitUsd);
469
- // eslint-disable-next-line no-param-reassign
470
- item.variableLimit = calculateBorrowingAssetLimit(item.borrowedUsdVariable, payload.borrowLimitUsd);
471
- // eslint-disable-next-line no-param-reassign
472
- item.limit = calculateBorrowingAssetLimit(item.borrowedUsd, payload.borrowLimitUsd);
473
- }
474
- });
475
-
476
- payload.isSubscribedToAutomation = false;
477
- payload.automationResubscribeRequired = false;
478
-
479
- return payload;
480
- };
481
-
482
- export const getAaveV3AccountData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress, extractedState: any, blockNumber: 'latest' | number = 'latest'): Promise<AaveV3PositionData> => _getAaveV3AccountData(getViemProvider(provider, network), network, address, extractedState, blockNumber);
483
-
484
- export const getAaveV3FullPositionData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress, market: AaveMarketInfo): Promise<AaveV3PositionData> => {
485
- const marketData = await getAaveV3MarketData(provider, network, market);
486
- const positionData = await getAaveV3AccountData(provider, network, address, { assetsData: marketData.assetsData, selectedMarket: market, eModeCategoriesData: marketData.eModeCategoriesData });
487
- return positionData;
488
- };
1
+ import { assetAmountInEth, assetAmountInWei, getAssetInfo } from '@defisaver/tokens';
2
+ import { Client } from 'viem';
3
+ import Dec from 'decimal.js';
4
+ import {
5
+ AaveIncentiveDataProviderV3ContractViem,
6
+ AaveIncentivesControllerViem,
7
+ AaveV3ViewContractViem,
8
+ createViemContractFromConfigFunc,
9
+ StkAAVEViem,
10
+ } from '../contracts';
11
+ import { aaveAnyGetAggregatedPositionData, aaveV3IsInIsolationMode, aaveV3IsInSiloedMode } from '../helpers/aaveHelpers';
12
+ import { AAVE_V3 } from '../markets/aave';
13
+ import { aprToApy, calculateBorrowingAssetLimit } from '../moneymarket';
14
+ import {
15
+ ethToWeth, isEnabledOnBitmap, isLayer2Network, wethToEth, wethToEthByAddress,
16
+ } from '../services/utils';
17
+ import { getStakingApy, STAKING_ASSETS } from '../staking';
18
+ import {
19
+ AaveMarketInfo,
20
+ AaveV3AssetData,
21
+ AaveV3AssetsData,
22
+ AaveV3MarketData,
23
+ AaveV3PositionData,
24
+ AaveV3UsedAsset,
25
+ AaveV3UsedAssets,
26
+ EModeCategoriesData,
27
+ EModeCategoryData,
28
+ EModeCategoryDataMapping,
29
+ } from '../types/aave';
30
+ import {
31
+ Blockish, EthAddress, EthereumProvider, HexString, NetworkNumber, PositionBalances,
32
+ } from '../types/common';
33
+ import { getViemProvider, setViemBlockNumber } from '../services/viem';
34
+ import { SECONDS_PER_DAY, SECONDS_PER_YEAR } from '../constants';
35
+
36
+ export const aaveV3EmodeCategoriesMapping = (extractedState: any, usedAssets: AaveV3UsedAssets) => {
37
+ const { eModeCategoriesData }: { assetsData: AaveV3AssetsData, eModeCategoriesData: EModeCategoriesData } = extractedState;
38
+ const usedAssetsValues = Object.values(usedAssets);
39
+
40
+ const categoriesMapping: { [key: number]: EModeCategoryDataMapping } = {};
41
+
42
+ Object.values(eModeCategoriesData).forEach((e: EModeCategoryData) => {
43
+ const borrowingOnlyFromCategory = e.id === 0
44
+ ? true
45
+ : !usedAssetsValues.filter(u => u.isBorrowed && !e.borrowAssets.includes(u.symbol)).length;
46
+ const afterEnteringCategory = aaveAnyGetAggregatedPositionData({
47
+ ...extractedState, usedAssets, eModeCategory: e.id,
48
+ });
49
+ const willStayOverCollateralized = new Dec(afterEnteringCategory.ratio).eq(0) || new Dec(afterEnteringCategory.ratio).gt(afterEnteringCategory.liqPercent);
50
+ const enteringTerms = [borrowingOnlyFromCategory, willStayOverCollateralized];
51
+ categoriesMapping[e.id] = {
52
+ enteringTerms,
53
+ canEnterCategory: !enteringTerms.includes(false),
54
+ id: e.id,
55
+ enabledData: {
56
+ ratio: afterEnteringCategory.ratio,
57
+ liqRatio: afterEnteringCategory.liqRatio,
58
+ liqPercent: afterEnteringCategory.liqPercent,
59
+ collRatio: afterEnteringCategory.collRatio,
60
+ },
61
+ };
62
+ });
63
+ return categoriesMapping;
64
+ };
65
+
66
+ export async function _getAaveV3MarketData(provider: Client, network: NetworkNumber, market: AaveMarketInfo, blockNumber: 'latest' | number = 'latest'): Promise<AaveV3MarketData> {
67
+ const _addresses = market.assets.map(a => getAssetInfo(ethToWeth(a), network).address);
68
+
69
+ const isL2 = isLayer2Network(network);
70
+ const loanInfoContract = AaveV3ViewContractViem(provider, network);
71
+ const aaveIncentivesContract = AaveIncentiveDataProviderV3ContractViem(provider, network);
72
+ const marketAddress = market.providerAddress;
73
+ const networksWithIncentives = [NetworkNumber.Eth, NetworkNumber.Arb, NetworkNumber.Opt, NetworkNumber.Linea];
74
+
75
+ // eslint-disable-next-line prefer-const
76
+ let [loanInfo, eModesInfo, isBorrowAllowed, rewardInfo] = await Promise.all([
77
+ loanInfoContract.read.getFullTokensInfo([marketAddress, _addresses as EthAddress[]], setViemBlockNumber(blockNumber)),
78
+ loanInfoContract.read.getAllEmodes([marketAddress], setViemBlockNumber(blockNumber)),
79
+ loanInfoContract.read.isBorrowAllowed([marketAddress], setViemBlockNumber(blockNumber)), // Used on L2s check for PriceOracleSentinel (mainnet will always return true)
80
+ networksWithIncentives.includes(network) ? aaveIncentivesContract.read.getReservesIncentivesData([marketAddress], setViemBlockNumber(blockNumber)) : null,
81
+ ]);
82
+ isBorrowAllowed = isLayer2Network(network) ? isBorrowAllowed : true;
83
+
84
+ const eModeCategoriesData: EModeCategoriesData = {};
85
+ for (let i = 0; i < eModesInfo.length; i++) {
86
+ if (!eModesInfo[i].label) break;
87
+ eModeCategoriesData[i + 1] = {
88
+ label: eModesInfo[i].label,
89
+ id: i + 1,
90
+ liquidationBonus: new Dec(eModesInfo[i].liquidationBonus).div(10000).toString(),
91
+ liquidationRatio: new Dec(eModesInfo[i].liquidationThreshold).div(10000).toString(),
92
+ collateralFactor: new Dec(eModesInfo[i].ltv).div(10000).toString(),
93
+ borrowableBitmap: eModesInfo[i].borrowableBitmap.toString(),
94
+ collateralBitmap: eModesInfo[i].collateralBitmap.toString(),
95
+ borrowAssets: [],
96
+ collateralAssets: [],
97
+ };
98
+ }
99
+
100
+ if (networksWithIncentives.includes(network) && rewardInfo) {
101
+ rewardInfo = rewardInfo.reduce((all: any, _market: any) => {
102
+ // eslint-disable-next-line no-param-reassign
103
+ all[_market.underlyingAsset] = _market;
104
+ return all;
105
+ }, {});
106
+ }
107
+
108
+ const assetsData: AaveV3AssetData[] = await Promise.all(loanInfo
109
+ .map(async (tokenMarket, i) => {
110
+ const symbol = market.assets[i];
111
+
112
+ // eslint-disable-next-line guard-for-in
113
+ for (const eModeIndex in eModeCategoriesData) {
114
+ if (isEnabledOnBitmap(Number(eModeCategoriesData[eModeIndex].collateralBitmap), Number(tokenMarket.assetId))) eModeCategoriesData[eModeIndex].collateralAssets.push(symbol);
115
+ if (isEnabledOnBitmap(Number(eModeCategoriesData[eModeIndex].borrowableBitmap), Number(tokenMarket.assetId))) eModeCategoriesData[eModeIndex].borrowAssets.push(symbol);
116
+ }
117
+
118
+ const borrowCap = tokenMarket.borrowCap.toString();
119
+
120
+ const borrowCapInWei = new Dec(assetAmountInWei(borrowCap, symbol));
121
+ let marketLiquidity = borrowCapInWei.lt(new Dec(tokenMarket.totalSupply.toString()))
122
+ ? assetAmountInEth(borrowCapInWei
123
+ .sub(tokenMarket.totalBorrow.toString())
124
+ .toString(), symbol)
125
+ : assetAmountInEth(new Dec(tokenMarket.totalSupply.toString())
126
+ .sub(tokenMarket.totalBorrow.toString())
127
+ .toString(), symbol);
128
+
129
+ if (new Dec(marketLiquidity).lt(0)) {
130
+ marketLiquidity = '0';
131
+ }
132
+ return ({
133
+ symbol,
134
+ isIsolated: new Dec(tokenMarket.debtCeilingForIsolationMode.toString()).gt(0),
135
+ debtCeilingForIsolationMode: new Dec(tokenMarket.debtCeilingForIsolationMode.toString()).div(100).toString(),
136
+ isSiloed: tokenMarket.isSiloedForBorrowing,
137
+ isolationModeTotalDebt: new Dec(tokenMarket.isolationModeTotalDebt.toString()).div(100).toString(),
138
+ assetId: Number(tokenMarket.assetId),
139
+ underlyingTokenAddress: tokenMarket.underlyingTokenAddress,
140
+ supplyRate: aprToApy(new Dec(tokenMarket.supplyRate.toString()).div(1e25).toString()),
141
+ borrowRate: aprToApy(new Dec(tokenMarket.borrowRateVariable.toString()).div(1e25).toString()),
142
+ borrowRateStable: aprToApy(new Dec(tokenMarket.borrowRateStable.toString()).div(1e25).toString()),
143
+ collateralFactor: new Dec(tokenMarket.collateralFactor.toString()).div(10000).toString(),
144
+ liquidationBonus: new Dec(tokenMarket.liquidationBonus.toString()).div(10000).toString(),
145
+ liquidationRatio: new Dec(tokenMarket.liquidationRatio.toString()).div(10000).toString(),
146
+ marketLiquidity,
147
+ utilization: new Dec(tokenMarket.totalBorrow.toString()).times(100).div(new Dec(tokenMarket.totalSupply.toString())).toString(),
148
+ usageAsCollateralEnabled: tokenMarket.usageAsCollateralEnabled,
149
+ supplyCap: tokenMarket.supplyCap.toString(),
150
+ borrowCap,
151
+ totalSupply: assetAmountInEth(tokenMarket.totalSupply.toString(), symbol),
152
+ isInactive: !tokenMarket.isActive,
153
+ isFrozen: tokenMarket.isFrozen,
154
+ isPaused: tokenMarket.isPaused,
155
+ canBeBorrowed: tokenMarket.isActive && !tokenMarket.isPaused && !tokenMarket.isFrozen && tokenMarket.borrowingEnabled && isBorrowAllowed,
156
+ canBeSupplied: tokenMarket.isActive && !tokenMarket.isPaused && !tokenMarket.isFrozen,
157
+ canBeWithdrawn: tokenMarket.isActive && !tokenMarket.isPaused,
158
+ canBePayBacked: tokenMarket.isActive && !tokenMarket.isPaused,
159
+ disabledStableBorrowing: !tokenMarket.stableBorrowRateEnabled,
160
+ totalBorrow: assetAmountInEth(tokenMarket.totalBorrow.toString(), symbol),
161
+ totalBorrowVar: assetAmountInEth(tokenMarket.totalBorrowVar.toString(), symbol),
162
+ price: new Dec(tokenMarket.price.toString()).div(1e8).toString(), // is actually price in USD
163
+ isolationModeBorrowingEnabled: tokenMarket.isolationModeBorrowingEnabled,
164
+ isFlashLoanEnabled: tokenMarket.isFlashLoanEnabled,
165
+ aTokenAddress: tokenMarket.aTokenAddress,
166
+ });
167
+ }));
168
+
169
+
170
+ // Get incentives data
171
+ await Promise.all(assetsData.map(async (_market: AaveV3AssetData) => {
172
+ /* eslint-disable no-param-reassign */
173
+ // @ts-ignore
174
+ const rewardForMarket = rewardInfo?.[_market.underlyingTokenAddress];
175
+ const isStakingAsset = STAKING_ASSETS.includes(_market.symbol);
176
+ if (isStakingAsset) {
177
+ _market.incentiveSupplyApy = await getStakingApy(_market.symbol);
178
+ _market.incentiveSupplyToken = _market.symbol;
179
+ if (!_market.supplyIncentives) {
180
+ _market.supplyIncentives = [];
181
+ }
182
+ _market.supplyIncentives.push({
183
+ apy: _market.incentiveSupplyApy || '0',
184
+ token: _market.symbol,
185
+ incentiveKind: 'staking',
186
+ });
187
+ }
188
+
189
+ if (_market.canBeBorrowed && _market.incentiveSupplyApy) {
190
+ _market.incentiveBorrowApy = _market.incentiveSupplyApy;
191
+ _market.incentiveBorrowToken = _market.incentiveSupplyToken;
192
+ if (!_market.borrowIncentives) {
193
+ _market.borrowIncentives = [];
194
+ }
195
+ _market.borrowIncentives.push({
196
+ apy: _market.incentiveBorrowApy,
197
+ token: _market.incentiveBorrowToken!!,
198
+ incentiveKind: 'reward',
199
+ });
200
+ }
201
+
202
+ if (_market.symbol === 'USDe') {
203
+ const merklApy = await getStakingApy(_market.symbol);
204
+ if (!_market.supplyIncentives) {
205
+ _market.supplyIncentives = [];
206
+ }
207
+ _market.supplyIncentives.push({
208
+ apy: merklApy || '0',
209
+ token: _market.symbol,
210
+ incentiveKind: 'reward',
211
+ });
212
+ }
213
+
214
+ if (!rewardForMarket) return;
215
+ // @ts-ignore
216
+ rewardForMarket.aIncentiveData.rewardsTokenInformation.forEach(supplyRewardData => {
217
+ if (supplyRewardData) {
218
+ if (+(supplyRewardData.emissionEndTimestamp.toString()) * 1000 < Date.now()) return;
219
+ _market.incentiveSupplyToken = supplyRewardData.rewardTokenSymbol;
220
+ // reward token is aave asset
221
+ if (supplyRewardData.rewardTokenSymbol.startsWith('a') && supplyRewardData.rewardTokenSymbol.includes(_market.symbol)) _market.incentiveSupplyToken = _market.symbol;
222
+ const supplyEmissionPerSecond = supplyRewardData.emissionPerSecond;
223
+ const supplyRewardPrice = new Dec(supplyRewardData.rewardPriceFeed).div(10 ** +supplyRewardData.priceFeedDecimals)
224
+ .toString();
225
+ const rewardApy = new Dec(supplyEmissionPerSecond).div((10 ** +supplyRewardData.rewardTokenDecimals) / 100)
226
+ .mul(365 * 24 * 3600)
227
+ .mul(supplyRewardPrice)
228
+ .div(_market.price)
229
+ .div(_market.totalSupply)
230
+ .toString();
231
+ _market.incentiveSupplyApy = new Dec(_market.incentiveSupplyApy || '0').add(rewardApy)
232
+ .toString();
233
+
234
+ if (!_market.supplyIncentives) {
235
+ _market.supplyIncentives = [];
236
+ }
237
+ _market.supplyIncentives.push({
238
+ token: supplyRewardData.rewardTokenSymbol,
239
+ apy: rewardApy,
240
+ incentiveKind: 'reward',
241
+ });
242
+ }
243
+ });
244
+ // @ts-ignore
245
+ rewardForMarket.vIncentiveData.rewardsTokenInformation.forEach(borrowRewardData => {
246
+ if (borrowRewardData) {
247
+ if (+(borrowRewardData.emissionEndTimestamp.toString()) * 1000 < Date.now()) return;
248
+ _market.incentiveBorrowToken = borrowRewardData.rewardTokenSymbol;
249
+ if (borrowRewardData.rewardTokenSymbol.startsWith('a') && borrowRewardData.rewardTokenSymbol.includes(_market.symbol)) _market.incentiveBorrowToken = _market.symbol;
250
+ const supplyEmissionPerSecond = borrowRewardData.emissionPerSecond;
251
+ const supplyRewardPrice = new Dec(borrowRewardData.rewardPriceFeed).div(10 ** +borrowRewardData.priceFeedDecimals)
252
+ .toString();
253
+ const rewardApy = new Dec(supplyEmissionPerSecond).div((10 ** +borrowRewardData.rewardTokenDecimals) / 100)
254
+ .mul(365 * 24 * 3600)
255
+ .mul(supplyRewardPrice)
256
+ .div(_market.price)
257
+ .div(_market.totalBorrowVar)
258
+ .toString();
259
+ _market.incentiveBorrowApy = new Dec(_market.incentiveBorrowApy || '0').add(rewardApy)
260
+ .toString();
261
+
262
+ if (!_market.borrowIncentives) {
263
+ _market.borrowIncentives = [];
264
+ }
265
+ _market.borrowIncentives.push({
266
+ token: borrowRewardData.rewardTokenSymbol,
267
+ apy: rewardApy,
268
+ incentiveKind: 'reward',
269
+ });
270
+ }
271
+ });
272
+ }));
273
+
274
+ const payload: AaveV3AssetsData = {};
275
+ // Sort by market size
276
+ assetsData
277
+ .sort((a, b) => {
278
+ const aMarket = new Dec(a.price).times(a.totalSupply).toString();
279
+ const bMarket = new Dec(b.price).times(b.totalSupply).toString();
280
+
281
+ return new Dec(bMarket).minus(aMarket).toNumber();
282
+ })
283
+ .forEach((assetData: AaveV3AssetData, i) => {
284
+ payload[assetData.symbol] = { ...assetData, sortIndex: i };
285
+ });
286
+
287
+ eModeCategoriesData[0] = {
288
+ id: 0,
289
+ label: '',
290
+ liquidationBonus: '0',
291
+ liquidationRatio: '0',
292
+ collateralFactor: '0',
293
+ collateralAssets: assetsData.map((a) => a.symbol),
294
+ borrowAssets: assetsData.map((a) => a.symbol),
295
+ };
296
+
297
+ return { assetsData: payload, eModeCategoriesData };
298
+ }
299
+
300
+ export async function getAaveV3MarketData(provider: EthereumProvider, network: NetworkNumber, market: AaveMarketInfo, blockNumber: 'latest' | number = 'latest'): Promise<AaveV3MarketData> {
301
+ return _getAaveV3MarketData(getViemProvider(provider, network), network, market, blockNumber);
302
+ }
303
+
304
+ export const EMPTY_AAVE_DATA = {
305
+ usedAssets: {},
306
+ suppliedUsd: '0',
307
+ borrowedUsd: '0',
308
+ borrowLimitUsd: '0',
309
+ leftToBorrowUsd: '0',
310
+ ratio: '0',
311
+ minRatio: '0',
312
+ netApy: '0',
313
+ incentiveUsd: '0',
314
+ totalInterestUsd: '0',
315
+ isSubscribedToAutomation: false,
316
+ automationResubscribeRequired: false,
317
+ eModeCategory: 0,
318
+ isInIsolationMode: false,
319
+ isInSiloedMode: false,
320
+ totalSupplied: '0',
321
+ eModeCategories: [],
322
+ collRatio: '0',
323
+ suppliedCollateralUsd: '0',
324
+ };
325
+
326
+ export const _getAaveV3AccountBalances = async (provider: Client, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress): Promise<PositionBalances> => {
327
+ let balances: PositionBalances = {
328
+ collateral: {},
329
+ debt: {},
330
+ };
331
+
332
+ if (!address) {
333
+ return balances;
334
+ }
335
+
336
+ const loanInfoContract = AaveV3ViewContractViem(provider, network, block);
337
+
338
+ const market = AAVE_V3(network);
339
+ const marketAddress = market.providerAddress;
340
+ // @ts-ignore
341
+ const protocolDataProviderContract = createViemContractFromConfigFunc(market.protocolData, market.protocolDataAddress)(provider, network);
342
+
343
+ // @ts-ignore
344
+ const reserveTokens = await protocolDataProviderContract.read.getAllReservesTokens(setViemBlockNumber(block)) as { symbol: string, tokenAddress: EthAddress }[];
345
+ const symbols = reserveTokens.map(({ symbol }: { symbol: string }) => symbol);
346
+ const _addresses = reserveTokens.map(({ tokenAddress }: { tokenAddress: EthAddress }) => tokenAddress);
347
+
348
+ // split addresses in half to avoid gas limit by multicall
349
+ const middleAddressIndex = Math.floor(_addresses.length / 2);
350
+
351
+ const [tokenBalances1, tokenBalances2] = await Promise.all([
352
+ loanInfoContract.read.getTokenBalances([marketAddress, address, _addresses.slice(0, middleAddressIndex)], setViemBlockNumber(block)),
353
+ loanInfoContract.read.getTokenBalances([marketAddress, address, _addresses.slice(middleAddressIndex, _addresses.length)], setViemBlockNumber(block)),
354
+ ]);
355
+
356
+ const loanInfo = [...tokenBalances1, ...tokenBalances2];
357
+
358
+ loanInfo.forEach((tokenInfo: any, i: number) => {
359
+ const asset = wethToEth(symbols[i]);
360
+ const assetAddr = wethToEthByAddress(_addresses[i], network).toLowerCase();
361
+
362
+ balances = {
363
+ collateral: {
364
+ ...balances.collateral,
365
+ [addressMapping ? assetAddr : asset]: tokenInfo.balance.toString(),
366
+ },
367
+ debt: {
368
+ ...balances.debt,
369
+ [addressMapping ? assetAddr : asset]: new Dec(tokenInfo.borrowsStable.toString()).add(tokenInfo.borrowsVariable.toString()).toString(),
370
+ },
371
+ };
372
+ });
373
+
374
+ return balances;
375
+ };
376
+
377
+ export const getAaveV3AccountBalances = async (provider: EthereumProvider, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress): Promise<PositionBalances> => _getAaveV3AccountBalances(getViemProvider(provider, network), network, block, addressMapping, address);
378
+
379
+ export const _getAaveV3AccountData = async (provider: Client, network: NetworkNumber, address: EthAddress, extractedState: any, blockNumber: 'latest' | number = 'latest'): Promise<AaveV3PositionData> => {
380
+ const {
381
+ selectedMarket: market, assetsData,
382
+ } = extractedState;
383
+ let payload: AaveV3PositionData = {
384
+ ...EMPTY_AAVE_DATA,
385
+ lastUpdated: Date.now(),
386
+ };
387
+ if (!address) {
388
+ return payload;
389
+ }
390
+
391
+ const loanInfoContract = AaveV3ViewContractViem(provider, network);
392
+ const lendingPoolContract = createViemContractFromConfigFunc(market.lendingPool, market.lendingPoolAddress)(provider, network);
393
+ const marketAddress = market.providerAddress;
394
+ const _addresses = market.assets.map((a: string[]) => getAssetInfo(ethToWeth(a), network).address);
395
+
396
+ const middleAddressIndex = Math.floor(_addresses.length / 2); // split addresses in half to avoid gas limit by multicall
397
+
398
+ const [eModeCategory, tokenBalances1, tokenBalances2] = await Promise.all([
399
+ lendingPoolContract.read.getUserEMode([address], setViemBlockNumber(blockNumber)),
400
+ loanInfoContract.read.getTokenBalances([marketAddress, address, _addresses.slice(0, middleAddressIndex) as EthAddress[]], setViemBlockNumber(blockNumber)),
401
+ loanInfoContract.read.getTokenBalances([marketAddress, address, _addresses.slice(middleAddressIndex, _addresses.length) as EthAddress[]], setViemBlockNumber(blockNumber)),
402
+ ]);
403
+
404
+ const loanInfo = [...tokenBalances1, ...tokenBalances2];
405
+
406
+ const usedAssets = {} as AaveV3UsedAssets;
407
+ loanInfo.map(async (tokenInfo, i) => {
408
+ const asset = market.assets[i];
409
+ const isSupplied = tokenInfo.balance.toString() !== '0';
410
+ const isBorrowed = tokenInfo.borrowsStable.toString() !== '0' || tokenInfo.borrowsVariable.toString() !== '0';
411
+ if (!isSupplied && !isBorrowed) return;
412
+
413
+ const supplied = assetAmountInEth(tokenInfo.balance.toString(), asset);
414
+ const borrowedStable = assetAmountInEth(tokenInfo.borrowsStable.toString(), asset);
415
+ const borrowedVariable = assetAmountInEth(tokenInfo.borrowsVariable.toString(), asset);
416
+ const enabledAsCollateral = assetsData[asset].usageAsCollateralEnabled ? tokenInfo.enabledAsCollateral : false;
417
+
418
+ let interestMode;
419
+ if (borrowedVariable === '0' && borrowedStable !== '0') {
420
+ interestMode = '1';
421
+ } else if (borrowedVariable !== '0' && borrowedStable === '0') {
422
+ interestMode = '2';
423
+ } else {
424
+ interestMode = 'both';
425
+ }
426
+ if (!usedAssets[asset]) usedAssets[asset] = {} as AaveV3UsedAsset;
427
+ const borrowed = new Dec(borrowedStable).add(borrowedVariable).toString();
428
+
429
+ usedAssets[asset] = {
430
+ ...usedAssets[asset],
431
+ symbol: asset,
432
+ supplied,
433
+ suppliedUsd: new Dec(supplied).mul(assetsData[asset].price).toString(),
434
+ isSupplied,
435
+ collateral: enabledAsCollateral,
436
+ stableBorrowRate: aprToApy(new Dec(tokenInfo.stableBorrowRate.toString()).div(1e25).toString()),
437
+ borrowedStable,
438
+ borrowedVariable,
439
+ borrowedUsdStable: new Dec(borrowedStable).mul(assetsData[asset].price).toString(),
440
+ borrowedUsdVariable: new Dec(borrowedVariable).mul(assetsData[asset].price).toString(),
441
+ borrowed,
442
+ borrowedUsd: new Dec(new Dec(borrowedVariable).add(borrowedStable)).mul(assetsData[asset].price).toString(),
443
+ isBorrowed,
444
+ interestMode,
445
+ };
446
+ });
447
+
448
+ payload.eModeCategory = +(eModeCategory as BigInt).toString();
449
+ payload = {
450
+ ...payload,
451
+ usedAssets,
452
+ ...aaveAnyGetAggregatedPositionData({
453
+ ...extractedState, usedAssets, eModeCategory: payload.eModeCategory,
454
+ }),
455
+ };
456
+ payload.eModeCategories = aaveV3EmodeCategoriesMapping(extractedState, usedAssets);
457
+ payload.isInIsolationMode = aaveV3IsInIsolationMode({ usedAssets, assetsData });
458
+ payload.isInSiloedMode = aaveV3IsInSiloedMode({ usedAssets, assetsData });
459
+
460
+ payload.ratio = payload.borrowedUsd && payload.borrowedUsd !== '0'
461
+ ? new Dec(payload.borrowLimitUsd).div(payload.borrowedUsd).mul(100).toString()
462
+ : '0';
463
+ payload.minRatio = '100';
464
+ payload.collRatio = payload.borrowedUsd && payload.borrowedUsd !== '0'
465
+ ? new Dec(payload.suppliedCollateralUsd).div(payload.borrowedUsd).mul(100).toString()
466
+ : '0';
467
+
468
+ // Calculate borrow limits per asset
469
+ Object.values(payload.usedAssets).forEach((item) => {
470
+ if (item.isBorrowed) {
471
+ // eslint-disable-next-line no-param-reassign
472
+ item.stableLimit = calculateBorrowingAssetLimit(item.borrowedUsdStable, payload.borrowLimitUsd);
473
+ // eslint-disable-next-line no-param-reassign
474
+ item.variableLimit = calculateBorrowingAssetLimit(item.borrowedUsdVariable, payload.borrowLimitUsd);
475
+ // eslint-disable-next-line no-param-reassign
476
+ item.limit = calculateBorrowingAssetLimit(item.borrowedUsd, payload.borrowLimitUsd);
477
+ }
478
+ });
479
+
480
+ payload.isSubscribedToAutomation = false;
481
+ payload.automationResubscribeRequired = false;
482
+
483
+ return payload;
484
+ };
485
+
486
+ export const getAaveV3AccountData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress, extractedState: any, blockNumber: 'latest' | number = 'latest'): Promise<AaveV3PositionData> => _getAaveV3AccountData(getViemProvider(provider, network), network, address, extractedState, blockNumber);
487
+
488
+ export const getAaveV3FullPositionData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress, market: AaveMarketInfo): Promise<AaveV3PositionData> => {
489
+ const marketData = await getAaveV3MarketData(provider, network, market);
490
+ const positionData = await getAaveV3AccountData(provider, network, address, { assetsData: marketData.assetsData, selectedMarket: market, eModeCategoriesData: marketData.eModeCategoriesData });
491
+ return positionData;
492
+ };
493
+
494
+ // aTokens eligible for AAVE rewards
495
+ export const REWARDABLE_ASSETS = [
496
+ '0x028171bCA77440897B824Ca71D1c56caC55b68A3', // DAI
497
+ '0x6C3c78838c761c6Ac7bE9F59fe808ea2A6E4379d',
498
+ '0xD37EE7e4f452C6638c96536e68090De8cBcdb583', // GUSD
499
+ '0x279AF5b99540c1A3A7E3CDd326e19659401eF99e',
500
+ '0xBcca60bB61934080951369a648Fb03DF4F96263C', // USDC
501
+ '0x619beb58998eD2278e08620f97007e1116D5D25b',
502
+ '0x3Ed3B47Dd13EC9a98b44e6204A523E766B225811', // USDT
503
+ '0x531842cEbbdD378f8ee36D171d6cC9C4fcf475Ec',
504
+ '0x9ff58f4fFB29fA2266Ab25e75e2A8b3503311656', // WBTC
505
+ '0x9c39809Dec7F95F5e0713634a4D0701329B3b4d2',
506
+ '0x030bA81f1c18d280636F32af80b9AAd02Cf0854e', // WETH
507
+ '0xF63B34710400CAd3e044cFfDcAb00a0f32E33eCf',
508
+ '0xa06bC25B5805d5F8d82847D191Cb4Af5A3e873E0', // LINK
509
+ '0x0b8f12b1788BFdE65Aa1ca52E3e9F3Ba401be16D',
510
+ '0x6C5024Cd4F8A59110119C56f8933403A539555EB', // SUSD
511
+ '0xdC6a3Ab17299D9C2A412B0e0a4C1f55446AE0817',
512
+ '0x5165d24277cD063F5ac44Efd447B27025e888f37', // YFI
513
+ '0x7EbD09022Be45AD993BAA1CEc61166Fcc8644d97',
514
+ '0xF256CC7847E919FAc9B808cC216cAc87CCF2f47a', // xSUSHI
515
+ '0xfAFEDF95E21184E3d880bd56D4806c4b8d31c69A',
516
+ '0xB9D7CB55f463405CDfBe4E90a6D2Df01C2B92BF1', // UNI
517
+ '0x5BdB050A92CADcCfCDcCCBFC17204a1C9cC0Ab73',
518
+ '0xc713e5E149D5D0715DcD1c156a020976e7E56B88', // MKR
519
+ '0xba728eAd5e496BE00DCF66F650b6d7758eCB50f8',
520
+ '0x101cc05f4A51C0319f570d5E146a8C625198e636', // TUSD
521
+ '0x01C0eb1f8c6F1C1bF74ae028697ce7AA2a8b0E92',
522
+ '0xc9BC48c72154ef3e5425641a3c747242112a46AF', // RAI
523
+ '0xB5385132EE8321977FfF44b60cDE9fE9AB0B4e6b',
524
+ '0x272F97b7a56a387aE942350bBC7Df5700f8a4576', // BAL
525
+ '0x13210D4Fe0d5402bd7Ecbc4B5bC5cFcA3b71adB0',
526
+ '0x2e8f4bdbe3d47d7d7de490437aea9915d930f1a3', // USDP
527
+ '0xfdb93b3b10936cf81fa59a02a7523b6e2149b2b7',
528
+ '0xA361718326c15715591c299427c62086F69923D9', // BUSD
529
+ '0xbA429f7011c9fa04cDd46a2Da24dc0FF0aC6099c',
530
+ '0xd4937682df3C8aEF4FE912A96A74121C0829E664', // FRAX
531
+ '0xfE8F19B17fFeF0fDbfe2671F248903055AFAA8Ca',
532
+ '0x683923dB55Fead99A79Fa01A27EeC3cB19679cC3', // FEI
533
+ '0xC2e10006AccAb7B45D9184FcF5b7EC7763f5BaAe',
534
+ '0x8dAE6Cb04688C62d939ed9B68d32Bc62e49970b1', // CRV
535
+ '0x00ad8eBF64F141f1C81e9f8f792d3d1631c6c684',
536
+ '0x6F634c6135D2EBD550000ac92F494F9CB8183dAe', // DPI
537
+ '0x4dDff5885a67E4EffeC55875a3977D7E60F82ae0',
538
+ ] as const;
539
+
540
+ export const fetchYearlyMeritApyForStakingGho = async () => {
541
+ try {
542
+ const response = await fetch('https://apps.aavechan.com/api/merit/aprs', { signal: AbortSignal.timeout(5000) });
543
+ const data = await response.json();
544
+ const apr = data?.currentAPR?.actionsAPR?.['ethereum-stkgho'] || '0' as string;
545
+ const apy = aprToApy(apr);
546
+ const apyWithDFSBonus = new Dec(apy).mul(1.05).toString(); // 5% bonus for DFS users
547
+ return apyWithDFSBonus;
548
+ } catch (e) {
549
+ const message = 'External API Failure: Failed to fetch yearly merit APY for staking GHO';
550
+ console.error(message, e);
551
+ return '0';
552
+ }
553
+ };
554
+
555
+ export const getStakeAaveData = async (provider: Client, network: NetworkNumber, address: EthAddress) => {
556
+ const stkGhoAddress = getAssetInfo('stkGHO').address as HexString;
557
+ const stkAaveAddress = getAssetInfo('stkAAVE').address as HexString;
558
+
559
+ const AaveIncentivesController = AaveIncentivesControllerViem(provider, network);
560
+ const stkAAVE = StkAAVEViem(provider, network);
561
+ const stkGHO = createViemContractFromConfigFunc('Erc20', stkGhoAddress as HexString)(provider, network);
562
+
563
+
564
+ const [aaveRewardsBalance, emissionsPerSecond, stkAAVEBalance, stkAAVETotalSupply, stkGHOBalance, ghoMeritApy] = await Promise.all([
565
+ AaveIncentivesController.read.getRewardsBalance([REWARDABLE_ASSETS, address]),
566
+ stkAAVE.read.assets([stkAaveAddress]),
567
+ stkAAVE.read.balanceOf([address]),
568
+ stkAAVE.read.totalSupply(),
569
+ stkGHO.read.balanceOf([address]),
570
+ fetchYearlyMeritApyForStakingGho(),
571
+ ]);
572
+
573
+
574
+ const stkAaveApy = new Dec(assetAmountInEth(emissionsPerSecond[0].toString(), 'GHO') || 0).mul(SECONDS_PER_YEAR).mul(100).div(assetAmountInEth(stkAAVETotalSupply.toString(), 'stkAAVE'))
575
+ .toString();
576
+ return {
577
+ activatedCooldown: '0',
578
+ activatedCooldownAmount: '0',
579
+ stkAaveRewardsBalance: '0',
580
+ aaveRewardsBalance: assetAmountInEth(aaveRewardsBalance.toString(), 'AAVE'),
581
+ stkAaveBalance: assetAmountInEth(stkAAVEBalance.toString(), 'stkAAVE'),
582
+ stkGhoBalance: assetAmountInEth(stkGHOBalance.toString(), 'GHO'),
583
+ ghoMeritApy,
584
+ stkAaveApy,
585
+ };
586
+ };