@defisaver/positions-sdk 2.1.30 → 2.1.31

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 (106) hide show
  1. package/.mocharc.json +4 -4
  2. package/.nvmrc +1 -1
  3. package/README.md +64 -64
  4. package/cjs/helpers/morphoBlueHelpers/index.js +66 -66
  5. package/cjs/markets/aave/marketAssets.js +2 -2
  6. package/cjs/savings/morphoVaults/index.js +17 -17
  7. package/cjs/staking/staking.js +3 -1
  8. package/esm/helpers/morphoBlueHelpers/index.js +66 -66
  9. package/esm/markets/aave/marketAssets.js +2 -2
  10. package/esm/savings/morphoVaults/index.js +17 -17
  11. package/esm/staking/staking.js +3 -1
  12. package/package.json +48 -48
  13. package/src/aaveV2/index.ts +240 -240
  14. package/src/aaveV3/index.ts +614 -614
  15. package/src/aaveV3/merit.ts +97 -97
  16. package/src/aaveV3/merkl.ts +74 -74
  17. package/src/claiming/aaveV3.ts +154 -154
  18. package/src/claiming/compV3.ts +22 -22
  19. package/src/claiming/ethena.ts +61 -61
  20. package/src/claiming/index.ts +12 -12
  21. package/src/claiming/king.ts +66 -66
  22. package/src/claiming/morphoBlue.ts +118 -118
  23. package/src/claiming/spark.ts +225 -225
  24. package/src/compoundV2/index.ts +244 -244
  25. package/src/compoundV3/index.ts +274 -274
  26. package/src/config/contracts.ts +1284 -1284
  27. package/src/constants/index.ts +10 -10
  28. package/src/contracts.ts +160 -160
  29. package/src/curveUsd/index.ts +254 -254
  30. package/src/eulerV2/index.ts +324 -324
  31. package/src/exchange/index.ts +25 -25
  32. package/src/fluid/index.ts +1800 -1800
  33. package/src/helpers/aaveHelpers/index.ts +187 -187
  34. package/src/helpers/compoundHelpers/index.ts +283 -283
  35. package/src/helpers/curveUsdHelpers/index.ts +40 -40
  36. package/src/helpers/eulerHelpers/index.ts +222 -222
  37. package/src/helpers/fluidHelpers/index.ts +326 -326
  38. package/src/helpers/index.ts +10 -10
  39. package/src/helpers/liquityV2Helpers/index.ts +82 -82
  40. package/src/helpers/llamaLendHelpers/index.ts +53 -53
  41. package/src/helpers/makerHelpers/index.ts +52 -52
  42. package/src/helpers/morphoBlueHelpers/index.ts +396 -396
  43. package/src/helpers/sparkHelpers/index.ts +158 -158
  44. package/src/index.ts +49 -49
  45. package/src/liquity/index.ts +159 -159
  46. package/src/liquityV2/index.ts +703 -703
  47. package/src/llamaLend/index.ts +305 -305
  48. package/src/maker/index.ts +223 -223
  49. package/src/markets/aave/index.ts +118 -118
  50. package/src/markets/aave/marketAssets.ts +54 -54
  51. package/src/markets/compound/index.ts +243 -243
  52. package/src/markets/compound/marketsAssets.ts +97 -97
  53. package/src/markets/curveUsd/index.ts +69 -69
  54. package/src/markets/euler/index.ts +26 -26
  55. package/src/markets/fluid/index.ts +2900 -2900
  56. package/src/markets/index.ts +25 -25
  57. package/src/markets/liquityV2/index.ts +102 -102
  58. package/src/markets/llamaLend/contractAddresses.ts +141 -141
  59. package/src/markets/llamaLend/index.ts +235 -235
  60. package/src/markets/morphoBlue/index.ts +971 -971
  61. package/src/markets/spark/index.ts +29 -29
  62. package/src/markets/spark/marketAssets.ts +12 -12
  63. package/src/moneymarket/moneymarketCommonService.ts +85 -85
  64. package/src/morphoBlue/index.ts +274 -274
  65. package/src/portfolio/index.ts +598 -598
  66. package/src/savings/index.ts +55 -55
  67. package/src/savings/makerDsr/index.ts +53 -53
  68. package/src/savings/makerDsr/options.ts +9 -9
  69. package/src/savings/morphoVaults/index.ts +80 -80
  70. package/src/savings/morphoVaults/options.ts +203 -203
  71. package/src/savings/sparkSavingsVaults/index.ts +60 -60
  72. package/src/savings/sparkSavingsVaults/options.ts +35 -35
  73. package/src/savings/yearnVaults/index.ts +73 -73
  74. package/src/savings/yearnVaults/options.ts +32 -32
  75. package/src/services/priceService.ts +278 -278
  76. package/src/services/utils.ts +115 -115
  77. package/src/services/viem.ts +34 -34
  78. package/src/setup.ts +8 -8
  79. package/src/spark/index.ts +456 -456
  80. package/src/staking/eligibility.ts +53 -53
  81. package/src/staking/index.ts +1 -1
  82. package/src/staking/staking.ts +187 -186
  83. package/src/types/aave.ts +196 -196
  84. package/src/types/claiming.ts +114 -114
  85. package/src/types/common.ts +107 -107
  86. package/src/types/compound.ts +144 -144
  87. package/src/types/curveUsd.ts +123 -123
  88. package/src/types/euler.ts +175 -175
  89. package/src/types/fluid.ts +483 -483
  90. package/src/types/index.ts +14 -14
  91. package/src/types/liquity.ts +30 -30
  92. package/src/types/liquityV2.ts +126 -126
  93. package/src/types/llamaLend.ts +159 -159
  94. package/src/types/maker.ts +63 -63
  95. package/src/types/merit.ts +1 -1
  96. package/src/types/merkl.ts +70 -70
  97. package/src/types/morphoBlue.ts +200 -200
  98. package/src/types/portfolio.ts +60 -60
  99. package/src/types/savings/index.ts +19 -19
  100. package/src/types/savings/makerDsr.ts +13 -13
  101. package/src/types/savings/morphoVaults.ts +33 -33
  102. package/src/types/savings/sparkSavingsVaults.ts +15 -15
  103. package/src/types/savings/yearnVaults.ts +14 -14
  104. package/src/types/spark.ts +133 -133
  105. package/src/umbrella/index.ts +69 -69
  106. package/src/umbrella/umbrellaUtils.ts +29 -29
package/package.json CHANGED
@@ -1,48 +1,48 @@
1
- {
2
- "name": "@defisaver/positions-sdk",
3
- "version": "2.1.30",
4
- "description": "",
5
- "main": "./cjs/index.js",
6
- "module": "./esm/index.js",
7
- "types": "./esm/index.d.ts",
8
- "scripts": {
9
- "build:esm": "rm -rf esm && tsc -p tsconfig.esm.json",
10
- "build:cjs": "rm -rf cjs && tsc -p tsconfig.cjs.json",
11
- "build": "npm run lint && npm run build:cjs && npm run build:esm",
12
- "dev": "tsc -p tsconfig.json --watch",
13
- "lint": "eslint src/ --fix",
14
- "lint-check": "eslint src/",
15
- "test": "mocha tests/*",
16
- "test-single": "mocha ./tests/$npm_config_name.ts",
17
- "test:debugger": "mocha --inspect-brk tests/*",
18
- "version-bump": "git commit -am \"Version bump to $(npm version patch | cut -c 2-)\""
19
- },
20
- "keywords": [],
21
- "author": "",
22
- "license": "ISC",
23
- "dependencies": {
24
- "@defisaver/tokens": "^1.7.13",
25
- "@types/lodash": "^4.17.15",
26
- "@types/memoizee": "^0.4.12",
27
- "decimal.js": "^10.6.0",
28
- "lodash": "^4.17.21",
29
- "memoizee": "^0.4.17",
30
- "viem": "^2.37.9",
31
- "graphql-request": "^1.8.2"
32
- },
33
- "devDependencies": {
34
- "@defisaver/eslint-config": "^1.0.1",
35
- "@metamask/eth-json-rpc-middleware": "^15.0.1",
36
- "@metamask/eth-json-rpc-provider": "^4.1.6",
37
- "@types/chai": "^5.0.0",
38
- "@types/mocha": "^10.0.9",
39
- "chai": "^4.3.8",
40
- "dotenv": "^16.3.1",
41
- "eslint": "^8.49.0",
42
- "eslint-plugin-import": "^2.31.0",
43
- "mocha": "^10.2.0",
44
- "nock": "^14.0.0",
45
- "ts-node": "^10.9.2",
46
- "typescript": "^5.2.2"
47
- }
48
- }
1
+ {
2
+ "name": "@defisaver/positions-sdk",
3
+ "version": "2.1.31",
4
+ "description": "",
5
+ "main": "./cjs/index.js",
6
+ "module": "./esm/index.js",
7
+ "types": "./esm/index.d.ts",
8
+ "scripts": {
9
+ "build:esm": "rm -rf esm && tsc -p tsconfig.esm.json",
10
+ "build:cjs": "rm -rf cjs && tsc -p tsconfig.cjs.json",
11
+ "build": "npm run lint && npm run build:cjs && npm run build:esm",
12
+ "dev": "tsc -p tsconfig.json --watch",
13
+ "lint": "eslint src/ --fix",
14
+ "lint-check": "eslint src/",
15
+ "test": "mocha tests/*",
16
+ "test-single": "mocha ./tests/$npm_config_name.ts",
17
+ "test:debugger": "mocha --inspect-brk tests/*",
18
+ "version-bump": "git commit -am \"Version bump to $(npm version patch | cut -c 2-)\""
19
+ },
20
+ "keywords": [],
21
+ "author": "",
22
+ "license": "ISC",
23
+ "dependencies": {
24
+ "@defisaver/tokens": "^1.7.15",
25
+ "@types/lodash": "^4.17.15",
26
+ "@types/memoizee": "^0.4.12",
27
+ "decimal.js": "^10.6.0",
28
+ "lodash": "^4.17.21",
29
+ "memoizee": "^0.4.17",
30
+ "viem": "^2.37.9",
31
+ "graphql-request": "^1.8.2"
32
+ },
33
+ "devDependencies": {
34
+ "@defisaver/eslint-config": "^1.0.1",
35
+ "@metamask/eth-json-rpc-middleware": "^15.0.1",
36
+ "@metamask/eth-json-rpc-provider": "^4.1.6",
37
+ "@types/chai": "^5.0.0",
38
+ "@types/mocha": "^10.0.9",
39
+ "chai": "^4.3.8",
40
+ "dotenv": "^16.3.1",
41
+ "eslint": "^8.49.0",
42
+ "eslint-plugin-import": "^2.31.0",
43
+ "mocha": "^10.2.0",
44
+ "nock": "^14.0.0",
45
+ "ts-node": "^10.9.2",
46
+ "typescript": "^5.2.2"
47
+ }
48
+ }
@@ -1,240 +1,240 @@
1
- import Dec from 'decimal.js';
2
- import { assetAmountInEth, getAssetInfo } from '@defisaver/tokens';
3
- import { Client } from 'viem';
4
- import {
5
- Blockish, EthAddress, EthereumProvider, IncentiveKind, NetworkNumber, PositionBalances,
6
- } from '../types/common';
7
- import { calculateNetApy, getStakingApy } from '../staking';
8
- import { ethToWeth, wethToEth, wethToEthByAddress } from '../services/utils';
9
- import { AaveLoanInfoV2ContractViem, createViemContractFromConfigFunc } from '../contracts';
10
- import { aprToApy, calculateBorrowingAssetLimit } from '../moneymarket';
11
- import {
12
- AaveMarketInfo, AaveV2AssetData, AaveV2AssetsData, AaveV2PositionData, AaveV2UsedAsset, AaveV2UsedAssets,
13
- } from '../types';
14
- import { EMPTY_AAVE_DATA } from '../aaveV3';
15
- import { AAVE_V2 } from '../markets/aave';
16
- import { aaveAnyGetAggregatedPositionData } from '../helpers/aaveHelpers';
17
- import { getEthPrice } from '../services/priceService';
18
- import { getViemProvider, setViemBlockNumber } from '../services/viem';
19
-
20
- export const _getAaveV2MarketsData = async (provider: Client, network: NetworkNumber, selectedMarket: AaveMarketInfo) => {
21
- const _addresses = selectedMarket.assets.map(a => getAssetInfo(ethToWeth(a)).address);
22
- const loanInfoContract = AaveLoanInfoV2ContractViem(provider, network);
23
- const marketAddress = selectedMarket.providerAddress;
24
- const [loanInfo, ethPrice] = await Promise.all([
25
- loanInfoContract.read.getFullTokensInfo([marketAddress, _addresses as EthAddress[]]),
26
- getEthPrice(provider),
27
- ]);
28
- const markets: AaveV2AssetData[] = loanInfo
29
- .map((market, i) => ({
30
- symbol: selectedMarket.assets[i],
31
- underlyingTokenAddress: market.underlyingTokenAddress,
32
- supplyRate: aprToApy(new Dec(market.supplyRate.toString()).div(1e25).toString()),
33
- borrowRate: aprToApy(new Dec(market.borrowRateVariable.toString()).div(1e25).toString()),
34
- borrowRateStable: aprToApy(new Dec(market.borrowRateStable.toString()).div(1e25).toString()),
35
- collateralFactor: new Dec(market.collateralFactor.toString()).div(10000).toString(),
36
- liquidationRatio: new Dec(market.liquidationRatio.toString()).div(10000).toString(),
37
- marketLiquidity: assetAmountInEth(new Dec(market.totalSupply.toString())
38
- .sub(market.totalBorrow.toString())
39
- .toString(), selectedMarket.assets[i]),
40
- utilization: new Dec(market.totalBorrow.toString())
41
- .div(new Dec(market.totalSupply.toString()))
42
- .times(100)
43
- .toString(),
44
- usageAsCollateralEnabled: market.usageAsCollateralEnabled,
45
- supplyCap: '0',
46
- borrowCap: '0', // v2 doesn't have borrow cap but adding it for compatability with v3
47
- totalSupply: assetAmountInEth(market.totalSupply.toString(), selectedMarket.assets[i]),
48
- isInactive: !market.isActive,
49
- isFrozen: market.isFrozen,
50
- canBeBorrowed: market.isActive && market.borrowingEnabled && !market.isFrozen,
51
- canBeSupplied: market.isActive && !market.isFrozen,
52
- canBeWithdrawn: market.isActive,
53
- canBePayBacked: market.isActive,
54
- disabledStableBorrowing: !market.stableBorrowRateEnabled,
55
- totalBorrow: assetAmountInEth(market.totalBorrow.toString(), selectedMarket.assets[i]),
56
- totalBorrowVar: assetAmountInEth(market.totalBorrowVar.toString(), selectedMarket.assets[i]),
57
- priceInEth: new Dec(market.price.toString()).div(1e18).toString(),
58
- supplyIncentives: [],
59
- borrowIncentives: [],
60
- price: new Dec(market.price.toString()).div(1e18).mul(ethPrice).toString(),
61
- }));
62
-
63
- const stEthMarket = markets.find(({ symbol }) => symbol === 'stETH');
64
- if (stEthMarket) {
65
- stEthMarket.supplyIncentives.push({
66
- apy: await getStakingApy('stETH'),
67
- token: 'stETH',
68
- incentiveKind: IncentiveKind.Staking,
69
- description: 'Native stETH yield.',
70
- });
71
- }
72
-
73
- const payload: AaveV2AssetsData = {};
74
- // Sort by market size
75
- markets
76
- .sort((a, b) => {
77
- const aMarket = new Dec(a.price).times(a.totalSupply).toString();
78
- const bMarket = new Dec(b.price).times(b.totalSupply).toString();
79
-
80
- return new Dec(bMarket).minus(aMarket).toNumber();
81
- })
82
- .forEach((assetData: AaveV2AssetData, i) => {
83
- payload[assetData.symbol] = { ...assetData, sortIndex: i };
84
- });
85
-
86
- return { assetsData: payload };
87
- };
88
-
89
- export const getAaveV2MarketsData = async (provider: EthereumProvider, network: NetworkNumber, selectedMarket: AaveMarketInfo) => _getAaveV2MarketsData(getViemProvider(provider, network), network, selectedMarket);
90
-
91
- export const _getAaveV2AccountBalances = async (provider: Client, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress): Promise<PositionBalances> => {
92
- let balances: PositionBalances = {
93
- collateral: {},
94
- debt: {},
95
- };
96
-
97
- if (!address) {
98
- return balances;
99
- }
100
-
101
- const market = AAVE_V2;
102
-
103
- const loanInfoContract = AaveLoanInfoV2ContractViem(provider, network, block);
104
-
105
- const marketAddress = market.providerAddress;
106
- // @ts-ignore
107
- const protocolDataProviderContract = createViemContractFromConfigFunc(market.protocolData, market.protocolDataAddress)(provider, network);
108
-
109
- // @ts-ignore
110
- const reserveTokens = await protocolDataProviderContract.read.getAllReservesTokens(setViemBlockNumber(block)) as { symbol: string, tokenAddress: EthAddress }[];
111
- const symbols = reserveTokens.map(({ symbol }: { symbol: string }) => symbol);
112
- const _addresses = reserveTokens.map(({ tokenAddress }: { tokenAddress: EthAddress }) => tokenAddress);
113
-
114
- const loanInfo = await loanInfoContract.read.getTokenBalances([marketAddress, address, _addresses as EthAddress[]], setViemBlockNumber(block));
115
-
116
- loanInfo.forEach((_tokenInfo: any, i: number) => {
117
- const asset = wethToEth(symbols[i]);
118
- const assetAddr = wethToEthByAddress(_addresses[i], network).toLowerCase();
119
- const tokenInfo = { ..._tokenInfo };
120
-
121
- // known bug: stETH leaves 1 wei on every transfer
122
- if (asset === 'stETH' && tokenInfo.balance.toString() === '1') {
123
- tokenInfo.balance = '0';
124
- }
125
-
126
- balances = {
127
- collateral: {
128
- ...balances.collateral,
129
- [addressMapping ? assetAddr : asset]: tokenInfo.balance.toString(),
130
- },
131
- debt: {
132
- ...balances.debt,
133
- [addressMapping ? assetAddr : asset]: new Dec(tokenInfo.borrowsStable.toString()).add(tokenInfo.borrowsVariable.toString()).toString(),
134
- },
135
- };
136
- });
137
-
138
- return balances;
139
- };
140
-
141
- export const getAaveV2AccountBalances = async (provider: EthereumProvider, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress): Promise<PositionBalances> => _getAaveV2AccountBalances(getViemProvider(provider, network), network, block, addressMapping, address);
142
-
143
- export const _getAaveV2AccountData = async (provider: Client, network: NetworkNumber, address: EthAddress, assetsData: AaveV2AssetsData, market: AaveMarketInfo): Promise<AaveV2PositionData> => {
144
- if (!address) throw new Error('Address is required');
145
-
146
- let payload: AaveV2PositionData = {
147
- ...EMPTY_AAVE_DATA,
148
- lastUpdated: Date.now(),
149
- };
150
-
151
- const loanInfoContract = AaveLoanInfoV2ContractViem(provider, network);
152
- const marketAddress = market.providerAddress;
153
- const _addresses = market.assets.map(a => getAssetInfo(ethToWeth(a)).address);
154
- const loanInfo = await loanInfoContract.read.getTokenBalances([marketAddress, address, _addresses as EthAddress[]]);
155
- const usedAssets = {} as AaveV2UsedAssets;
156
- loanInfo.forEach((_tokenInfo, i) => {
157
- const asset = market.assets[i];
158
- const tokenInfo = { ..._tokenInfo };
159
-
160
- // known bug: stETH leaves 1 wei on every transfer
161
- if (asset === 'stETH' && tokenInfo.balance.toString() === '1') tokenInfo.balance = BigInt('0');
162
-
163
- const isSupplied = tokenInfo.balance.toString() !== '0';
164
- const isBorrowed = tokenInfo.borrowsStable.toString() !== '0' || tokenInfo.borrowsVariable.toString() !== '0';
165
- if (!isSupplied && !isBorrowed) return;
166
-
167
- const supplied = assetAmountInEth(tokenInfo.balance.toString(), asset);
168
- const borrowedStable = assetAmountInEth(tokenInfo.borrowsStable.toString(), asset);
169
- const borrowedVariable = assetAmountInEth(tokenInfo.borrowsVariable.toString(), asset);
170
- const enabledAsCollateral = assetsData[asset].usageAsCollateralEnabled ? tokenInfo.enabledAsCollateral : false;
171
- let interestMode;
172
- if (borrowedVariable === '0' && borrowedStable !== '0') {
173
- interestMode = '1';
174
- } else if (borrowedVariable !== '0' && borrowedStable === '0') {
175
- interestMode = '2';
176
- } else {
177
- interestMode = 'both';
178
- }
179
- if (!usedAssets[asset]) usedAssets[asset] = {} as AaveV2UsedAsset;
180
- usedAssets[asset] = {
181
- ...usedAssets[asset],
182
- symbol: asset,
183
- supplied,
184
- suppliedUsd: new Dec(supplied).mul(assetsData[asset].price).toString(),
185
- isSupplied,
186
- collateral: enabledAsCollateral,
187
- stableBorrowRate: aprToApy(new Dec(tokenInfo.stableBorrowRate).div(1e25).toString()),
188
- borrowedStable,
189
- borrowedVariable,
190
- borrowedUsdStable: new Dec(borrowedStable).mul(assetsData[asset].price).toString(),
191
- borrowedUsdVariable: new Dec(borrowedVariable).mul(assetsData[asset].price).toString(),
192
- borrowed: new Dec(borrowedStable).add(borrowedVariable).toString(),
193
- borrowedUsd: new Dec(new Dec(borrowedVariable).add(borrowedStable)).mul(assetsData[asset].price).toString(),
194
- isBorrowed,
195
- interestMode,
196
- };
197
- });
198
- payload = {
199
- ...payload,
200
- usedAssets,
201
- ...aaveAnyGetAggregatedPositionData({
202
- usedAssets, assetsData, eModeCategory: 0, selectedMarket: market,
203
- }),
204
- };
205
-
206
- payload.ratio = payload.borrowedUsd && payload.borrowedUsd !== '0'
207
- ? new Dec(payload.borrowLimitUsd).div(payload.borrowedUsd).mul(100).toString()
208
- : '0';
209
- payload.minRatio = '100';
210
- payload.collRatio = payload.borrowedUsd && payload.borrowedUsd !== '0'
211
- ? new Dec(payload.suppliedCollateralUsd).div(payload.borrowedUsd).mul(100).toString()
212
- : '0';
213
-
214
- // Calculate borrow limits per asset
215
- Object.values(payload.usedAssets).forEach((item) => {
216
- if (item.isBorrowed) {
217
- // eslint-disable-next-line no-param-reassign
218
- item.stableLimit = calculateBorrowingAssetLimit(item.borrowedUsdStable, payload.borrowLimitUsd);
219
- // eslint-disable-next-line no-param-reassign
220
- item.variableLimit = calculateBorrowingAssetLimit(item.borrowedUsdVariable, payload.borrowLimitUsd);
221
- // eslint-disable-next-line no-param-reassign
222
- item.limit = calculateBorrowingAssetLimit(item.borrowedUsd, payload.borrowLimitUsd);
223
- }
224
- });
225
-
226
- const { netApy, incentiveUsd, totalInterestUsd } = calculateNetApy({ usedAssets, assetsData });
227
- payload.netApy = netApy;
228
- payload.incentiveUsd = incentiveUsd;
229
- payload.totalInterestUsd = totalInterestUsd;
230
-
231
- return payload;
232
- };
233
-
234
- export const getAaveV2AccountData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress, assetsData: AaveV2AssetsData, market: AaveMarketInfo): Promise<AaveV2PositionData> => _getAaveV2AccountData(getViemProvider(provider, network), network, address, assetsData, market);
235
-
236
- export const getAaveV2FullPositionData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress, market: AaveMarketInfo): Promise<AaveV2PositionData> => {
237
- const marketData = await getAaveV2MarketsData(provider, network, market);
238
- const positionData = await getAaveV2AccountData(provider, network, address, marketData.assetsData, market);
239
- return positionData;
240
- };
1
+ import Dec from 'decimal.js';
2
+ import { assetAmountInEth, getAssetInfo } from '@defisaver/tokens';
3
+ import { Client } from 'viem';
4
+ import {
5
+ Blockish, EthAddress, EthereumProvider, IncentiveKind, NetworkNumber, PositionBalances,
6
+ } from '../types/common';
7
+ import { calculateNetApy, getStakingApy } from '../staking';
8
+ import { ethToWeth, wethToEth, wethToEthByAddress } from '../services/utils';
9
+ import { AaveLoanInfoV2ContractViem, createViemContractFromConfigFunc } from '../contracts';
10
+ import { aprToApy, calculateBorrowingAssetLimit } from '../moneymarket';
11
+ import {
12
+ AaveMarketInfo, AaveV2AssetData, AaveV2AssetsData, AaveV2PositionData, AaveV2UsedAsset, AaveV2UsedAssets,
13
+ } from '../types';
14
+ import { EMPTY_AAVE_DATA } from '../aaveV3';
15
+ import { AAVE_V2 } from '../markets/aave';
16
+ import { aaveAnyGetAggregatedPositionData } from '../helpers/aaveHelpers';
17
+ import { getEthPrice } from '../services/priceService';
18
+ import { getViemProvider, setViemBlockNumber } from '../services/viem';
19
+
20
+ export const _getAaveV2MarketsData = async (provider: Client, network: NetworkNumber, selectedMarket: AaveMarketInfo) => {
21
+ const _addresses = selectedMarket.assets.map(a => getAssetInfo(ethToWeth(a)).address);
22
+ const loanInfoContract = AaveLoanInfoV2ContractViem(provider, network);
23
+ const marketAddress = selectedMarket.providerAddress;
24
+ const [loanInfo, ethPrice] = await Promise.all([
25
+ loanInfoContract.read.getFullTokensInfo([marketAddress, _addresses as EthAddress[]]),
26
+ getEthPrice(provider),
27
+ ]);
28
+ const markets: AaveV2AssetData[] = loanInfo
29
+ .map((market, i) => ({
30
+ symbol: selectedMarket.assets[i],
31
+ underlyingTokenAddress: market.underlyingTokenAddress,
32
+ supplyRate: aprToApy(new Dec(market.supplyRate.toString()).div(1e25).toString()),
33
+ borrowRate: aprToApy(new Dec(market.borrowRateVariable.toString()).div(1e25).toString()),
34
+ borrowRateStable: aprToApy(new Dec(market.borrowRateStable.toString()).div(1e25).toString()),
35
+ collateralFactor: new Dec(market.collateralFactor.toString()).div(10000).toString(),
36
+ liquidationRatio: new Dec(market.liquidationRatio.toString()).div(10000).toString(),
37
+ marketLiquidity: assetAmountInEth(new Dec(market.totalSupply.toString())
38
+ .sub(market.totalBorrow.toString())
39
+ .toString(), selectedMarket.assets[i]),
40
+ utilization: new Dec(market.totalBorrow.toString())
41
+ .div(new Dec(market.totalSupply.toString()))
42
+ .times(100)
43
+ .toString(),
44
+ usageAsCollateralEnabled: market.usageAsCollateralEnabled,
45
+ supplyCap: '0',
46
+ borrowCap: '0', // v2 doesn't have borrow cap but adding it for compatability with v3
47
+ totalSupply: assetAmountInEth(market.totalSupply.toString(), selectedMarket.assets[i]),
48
+ isInactive: !market.isActive,
49
+ isFrozen: market.isFrozen,
50
+ canBeBorrowed: market.isActive && market.borrowingEnabled && !market.isFrozen,
51
+ canBeSupplied: market.isActive && !market.isFrozen,
52
+ canBeWithdrawn: market.isActive,
53
+ canBePayBacked: market.isActive,
54
+ disabledStableBorrowing: !market.stableBorrowRateEnabled,
55
+ totalBorrow: assetAmountInEth(market.totalBorrow.toString(), selectedMarket.assets[i]),
56
+ totalBorrowVar: assetAmountInEth(market.totalBorrowVar.toString(), selectedMarket.assets[i]),
57
+ priceInEth: new Dec(market.price.toString()).div(1e18).toString(),
58
+ supplyIncentives: [],
59
+ borrowIncentives: [],
60
+ price: new Dec(market.price.toString()).div(1e18).mul(ethPrice).toString(),
61
+ }));
62
+
63
+ const stEthMarket = markets.find(({ symbol }) => symbol === 'stETH');
64
+ if (stEthMarket) {
65
+ stEthMarket.supplyIncentives.push({
66
+ apy: await getStakingApy('stETH'),
67
+ token: 'stETH',
68
+ incentiveKind: IncentiveKind.Staking,
69
+ description: 'Native stETH yield.',
70
+ });
71
+ }
72
+
73
+ const payload: AaveV2AssetsData = {};
74
+ // Sort by market size
75
+ markets
76
+ .sort((a, b) => {
77
+ const aMarket = new Dec(a.price).times(a.totalSupply).toString();
78
+ const bMarket = new Dec(b.price).times(b.totalSupply).toString();
79
+
80
+ return new Dec(bMarket).minus(aMarket).toNumber();
81
+ })
82
+ .forEach((assetData: AaveV2AssetData, i) => {
83
+ payload[assetData.symbol] = { ...assetData, sortIndex: i };
84
+ });
85
+
86
+ return { assetsData: payload };
87
+ };
88
+
89
+ export const getAaveV2MarketsData = async (provider: EthereumProvider, network: NetworkNumber, selectedMarket: AaveMarketInfo) => _getAaveV2MarketsData(getViemProvider(provider, network), network, selectedMarket);
90
+
91
+ export const _getAaveV2AccountBalances = async (provider: Client, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress): Promise<PositionBalances> => {
92
+ let balances: PositionBalances = {
93
+ collateral: {},
94
+ debt: {},
95
+ };
96
+
97
+ if (!address) {
98
+ return balances;
99
+ }
100
+
101
+ const market = AAVE_V2;
102
+
103
+ const loanInfoContract = AaveLoanInfoV2ContractViem(provider, network, block);
104
+
105
+ const marketAddress = market.providerAddress;
106
+ // @ts-ignore
107
+ const protocolDataProviderContract = createViemContractFromConfigFunc(market.protocolData, market.protocolDataAddress)(provider, network);
108
+
109
+ // @ts-ignore
110
+ const reserveTokens = await protocolDataProviderContract.read.getAllReservesTokens(setViemBlockNumber(block)) as { symbol: string, tokenAddress: EthAddress }[];
111
+ const symbols = reserveTokens.map(({ symbol }: { symbol: string }) => symbol);
112
+ const _addresses = reserveTokens.map(({ tokenAddress }: { tokenAddress: EthAddress }) => tokenAddress);
113
+
114
+ const loanInfo = await loanInfoContract.read.getTokenBalances([marketAddress, address, _addresses as EthAddress[]], setViemBlockNumber(block));
115
+
116
+ loanInfo.forEach((_tokenInfo: any, i: number) => {
117
+ const asset = wethToEth(symbols[i]);
118
+ const assetAddr = wethToEthByAddress(_addresses[i], network).toLowerCase();
119
+ const tokenInfo = { ..._tokenInfo };
120
+
121
+ // known bug: stETH leaves 1 wei on every transfer
122
+ if (asset === 'stETH' && tokenInfo.balance.toString() === '1') {
123
+ tokenInfo.balance = '0';
124
+ }
125
+
126
+ balances = {
127
+ collateral: {
128
+ ...balances.collateral,
129
+ [addressMapping ? assetAddr : asset]: tokenInfo.balance.toString(),
130
+ },
131
+ debt: {
132
+ ...balances.debt,
133
+ [addressMapping ? assetAddr : asset]: new Dec(tokenInfo.borrowsStable.toString()).add(tokenInfo.borrowsVariable.toString()).toString(),
134
+ },
135
+ };
136
+ });
137
+
138
+ return balances;
139
+ };
140
+
141
+ export const getAaveV2AccountBalances = async (provider: EthereumProvider, network: NetworkNumber, block: Blockish, addressMapping: boolean, address: EthAddress): Promise<PositionBalances> => _getAaveV2AccountBalances(getViemProvider(provider, network), network, block, addressMapping, address);
142
+
143
+ export const _getAaveV2AccountData = async (provider: Client, network: NetworkNumber, address: EthAddress, assetsData: AaveV2AssetsData, market: AaveMarketInfo): Promise<AaveV2PositionData> => {
144
+ if (!address) throw new Error('Address is required');
145
+
146
+ let payload: AaveV2PositionData = {
147
+ ...EMPTY_AAVE_DATA,
148
+ lastUpdated: Date.now(),
149
+ };
150
+
151
+ const loanInfoContract = AaveLoanInfoV2ContractViem(provider, network);
152
+ const marketAddress = market.providerAddress;
153
+ const _addresses = market.assets.map(a => getAssetInfo(ethToWeth(a)).address);
154
+ const loanInfo = await loanInfoContract.read.getTokenBalances([marketAddress, address, _addresses as EthAddress[]]);
155
+ const usedAssets = {} as AaveV2UsedAssets;
156
+ loanInfo.forEach((_tokenInfo, i) => {
157
+ const asset = market.assets[i];
158
+ const tokenInfo = { ..._tokenInfo };
159
+
160
+ // known bug: stETH leaves 1 wei on every transfer
161
+ if (asset === 'stETH' && tokenInfo.balance.toString() === '1') tokenInfo.balance = BigInt('0');
162
+
163
+ const isSupplied = tokenInfo.balance.toString() !== '0';
164
+ const isBorrowed = tokenInfo.borrowsStable.toString() !== '0' || tokenInfo.borrowsVariable.toString() !== '0';
165
+ if (!isSupplied && !isBorrowed) return;
166
+
167
+ const supplied = assetAmountInEth(tokenInfo.balance.toString(), asset);
168
+ const borrowedStable = assetAmountInEth(tokenInfo.borrowsStable.toString(), asset);
169
+ const borrowedVariable = assetAmountInEth(tokenInfo.borrowsVariable.toString(), asset);
170
+ const enabledAsCollateral = assetsData[asset].usageAsCollateralEnabled ? tokenInfo.enabledAsCollateral : false;
171
+ let interestMode;
172
+ if (borrowedVariable === '0' && borrowedStable !== '0') {
173
+ interestMode = '1';
174
+ } else if (borrowedVariable !== '0' && borrowedStable === '0') {
175
+ interestMode = '2';
176
+ } else {
177
+ interestMode = 'both';
178
+ }
179
+ if (!usedAssets[asset]) usedAssets[asset] = {} as AaveV2UsedAsset;
180
+ usedAssets[asset] = {
181
+ ...usedAssets[asset],
182
+ symbol: asset,
183
+ supplied,
184
+ suppliedUsd: new Dec(supplied).mul(assetsData[asset].price).toString(),
185
+ isSupplied,
186
+ collateral: enabledAsCollateral,
187
+ stableBorrowRate: aprToApy(new Dec(tokenInfo.stableBorrowRate).div(1e25).toString()),
188
+ borrowedStable,
189
+ borrowedVariable,
190
+ borrowedUsdStable: new Dec(borrowedStable).mul(assetsData[asset].price).toString(),
191
+ borrowedUsdVariable: new Dec(borrowedVariable).mul(assetsData[asset].price).toString(),
192
+ borrowed: new Dec(borrowedStable).add(borrowedVariable).toString(),
193
+ borrowedUsd: new Dec(new Dec(borrowedVariable).add(borrowedStable)).mul(assetsData[asset].price).toString(),
194
+ isBorrowed,
195
+ interestMode,
196
+ };
197
+ });
198
+ payload = {
199
+ ...payload,
200
+ usedAssets,
201
+ ...aaveAnyGetAggregatedPositionData({
202
+ usedAssets, assetsData, eModeCategory: 0, selectedMarket: market,
203
+ }),
204
+ };
205
+
206
+ payload.ratio = payload.borrowedUsd && payload.borrowedUsd !== '0'
207
+ ? new Dec(payload.borrowLimitUsd).div(payload.borrowedUsd).mul(100).toString()
208
+ : '0';
209
+ payload.minRatio = '100';
210
+ payload.collRatio = payload.borrowedUsd && payload.borrowedUsd !== '0'
211
+ ? new Dec(payload.suppliedCollateralUsd).div(payload.borrowedUsd).mul(100).toString()
212
+ : '0';
213
+
214
+ // Calculate borrow limits per asset
215
+ Object.values(payload.usedAssets).forEach((item) => {
216
+ if (item.isBorrowed) {
217
+ // eslint-disable-next-line no-param-reassign
218
+ item.stableLimit = calculateBorrowingAssetLimit(item.borrowedUsdStable, payload.borrowLimitUsd);
219
+ // eslint-disable-next-line no-param-reassign
220
+ item.variableLimit = calculateBorrowingAssetLimit(item.borrowedUsdVariable, payload.borrowLimitUsd);
221
+ // eslint-disable-next-line no-param-reassign
222
+ item.limit = calculateBorrowingAssetLimit(item.borrowedUsd, payload.borrowLimitUsd);
223
+ }
224
+ });
225
+
226
+ const { netApy, incentiveUsd, totalInterestUsd } = calculateNetApy({ usedAssets, assetsData });
227
+ payload.netApy = netApy;
228
+ payload.incentiveUsd = incentiveUsd;
229
+ payload.totalInterestUsd = totalInterestUsd;
230
+
231
+ return payload;
232
+ };
233
+
234
+ export const getAaveV2AccountData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress, assetsData: AaveV2AssetsData, market: AaveMarketInfo): Promise<AaveV2PositionData> => _getAaveV2AccountData(getViemProvider(provider, network), network, address, assetsData, market);
235
+
236
+ export const getAaveV2FullPositionData = async (provider: EthereumProvider, network: NetworkNumber, address: EthAddress, market: AaveMarketInfo): Promise<AaveV2PositionData> => {
237
+ const marketData = await getAaveV2MarketsData(provider, network, market);
238
+ const positionData = await getAaveV2AccountData(provider, network, address, marketData.assetsData, market);
239
+ return positionData;
240
+ };