@defisaver/positions-sdk 2.1.43-dev-stats-test-dev → 2.1.43

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 (116) hide show
  1. package/.mocharc.json +4 -4
  2. package/.nvmrc +1 -1
  3. package/README.md +64 -64
  4. package/cjs/aaveV3/index.js +0 -13
  5. package/cjs/aaveV3/merit.js +0 -1
  6. package/cjs/aaveV3/merkl.js +0 -1
  7. package/cjs/helpers/morphoBlueHelpers/index.js +66 -66
  8. package/cjs/savings/morphoVaults/index.js +17 -17
  9. package/cjs/savings/morphoVaults/options.js +1 -1
  10. package/esm/aaveV3/index.js +0 -13
  11. package/esm/aaveV3/merit.js +0 -1
  12. package/esm/aaveV3/merkl.js +0 -1
  13. package/esm/helpers/morphoBlueHelpers/index.js +66 -66
  14. package/esm/savings/morphoVaults/index.js +17 -17
  15. package/esm/savings/morphoVaults/options.js +1 -1
  16. package/package.json +48 -48
  17. package/src/aaveV2/index.ts +240 -240
  18. package/src/aaveV3/index.ts +625 -634
  19. package/src/aaveV3/merit.ts +97 -98
  20. package/src/aaveV3/merkl.ts +74 -75
  21. package/src/claiming/aaveV3.ts +154 -154
  22. package/src/claiming/compV3.ts +22 -22
  23. package/src/claiming/ethena.ts +61 -61
  24. package/src/claiming/index.ts +12 -12
  25. package/src/claiming/king.ts +66 -66
  26. package/src/claiming/morphoBlue.ts +118 -118
  27. package/src/claiming/spark.ts +225 -225
  28. package/src/compoundV2/index.ts +244 -244
  29. package/src/compoundV3/index.ts +274 -274
  30. package/src/config/contracts.ts +1295 -1295
  31. package/src/constants/index.ts +10 -10
  32. package/src/contracts.ts +171 -171
  33. package/src/curveUsd/index.ts +254 -254
  34. package/src/eulerV2/index.ts +324 -324
  35. package/src/exchange/index.ts +25 -25
  36. package/src/fluid/index.ts +1800 -1800
  37. package/src/helpers/aaveHelpers/index.ts +187 -187
  38. package/src/helpers/compoundHelpers/index.ts +283 -283
  39. package/src/helpers/curveUsdHelpers/index.ts +40 -40
  40. package/src/helpers/eulerHelpers/index.ts +222 -222
  41. package/src/helpers/fluidHelpers/index.ts +326 -326
  42. package/src/helpers/index.ts +10 -10
  43. package/src/helpers/liquityV2Helpers/index.ts +82 -82
  44. package/src/helpers/llamaLendHelpers/index.ts +53 -53
  45. package/src/helpers/makerHelpers/index.ts +52 -52
  46. package/src/helpers/morphoBlueHelpers/index.ts +396 -396
  47. package/src/helpers/sparkHelpers/index.ts +158 -158
  48. package/src/index.ts +49 -49
  49. package/src/liquity/index.ts +159 -159
  50. package/src/liquityV2/index.ts +703 -703
  51. package/src/llamaLend/index.ts +305 -305
  52. package/src/maker/index.ts +223 -223
  53. package/src/markets/aave/index.ts +118 -118
  54. package/src/markets/aave/marketAssets.ts +54 -54
  55. package/src/markets/compound/index.ts +243 -243
  56. package/src/markets/compound/marketsAssets.ts +97 -97
  57. package/src/markets/curveUsd/index.ts +69 -69
  58. package/src/markets/euler/index.ts +26 -26
  59. package/src/markets/fluid/index.ts +2900 -2900
  60. package/src/markets/index.ts +25 -25
  61. package/src/markets/liquityV2/index.ts +102 -102
  62. package/src/markets/llamaLend/contractAddresses.ts +141 -141
  63. package/src/markets/llamaLend/index.ts +235 -235
  64. package/src/markets/morphoBlue/index.ts +971 -971
  65. package/src/markets/spark/index.ts +29 -29
  66. package/src/markets/spark/marketAssets.ts +12 -12
  67. package/src/moneymarket/moneymarketCommonService.ts +85 -85
  68. package/src/morphoBlue/index.ts +274 -274
  69. package/src/portfolio/index.ts +598 -598
  70. package/src/savings/index.ts +95 -95
  71. package/src/savings/makerDsr/index.ts +53 -53
  72. package/src/savings/makerDsr/options.ts +9 -9
  73. package/src/savings/morphoVaults/index.ts +80 -80
  74. package/src/savings/morphoVaults/options.ts +193 -193
  75. package/src/savings/skyOptions/index.ts +95 -95
  76. package/src/savings/skyOptions/options.ts +10 -10
  77. package/src/savings/sparkSavingsVaults/index.ts +60 -60
  78. package/src/savings/sparkSavingsVaults/options.ts +35 -35
  79. package/src/savings/yearnV3Vaults/index.ts +61 -61
  80. package/src/savings/yearnV3Vaults/options.ts +55 -55
  81. package/src/savings/yearnVaults/index.ts +73 -73
  82. package/src/savings/yearnVaults/options.ts +32 -32
  83. package/src/services/priceService.ts +278 -278
  84. package/src/services/utils.ts +115 -115
  85. package/src/services/viem.ts +34 -34
  86. package/src/setup.ts +8 -8
  87. package/src/spark/index.ts +458 -458
  88. package/src/staking/eligibility.ts +53 -53
  89. package/src/staking/index.ts +1 -1
  90. package/src/staking/staking.ts +186 -186
  91. package/src/types/aave.ts +196 -196
  92. package/src/types/claiming.ts +114 -114
  93. package/src/types/common.ts +107 -107
  94. package/src/types/compound.ts +144 -144
  95. package/src/types/curveUsd.ts +123 -123
  96. package/src/types/euler.ts +175 -175
  97. package/src/types/fluid.ts +483 -483
  98. package/src/types/index.ts +14 -14
  99. package/src/types/liquity.ts +30 -30
  100. package/src/types/liquityV2.ts +126 -126
  101. package/src/types/llamaLend.ts +159 -159
  102. package/src/types/maker.ts +63 -63
  103. package/src/types/merit.ts +1 -1
  104. package/src/types/merkl.ts +70 -70
  105. package/src/types/morphoBlue.ts +200 -200
  106. package/src/types/portfolio.ts +60 -60
  107. package/src/types/savings/index.ts +23 -23
  108. package/src/types/savings/makerDsr.ts +13 -13
  109. package/src/types/savings/morphoVaults.ts +32 -32
  110. package/src/types/savings/sky.ts +14 -14
  111. package/src/types/savings/sparkSavingsVaults.ts +15 -15
  112. package/src/types/savings/yearnV3Vaults.ts +17 -17
  113. package/src/types/savings/yearnVaults.ts +14 -14
  114. package/src/types/spark.ts +134 -134
  115. package/src/umbrella/index.ts +69 -69
  116. package/src/umbrella/umbrellaUtils.ts +29 -29
@@ -1,226 +1,226 @@
1
- import { assetAmountInEth, getAssetInfoByAddress } from '@defisaver/tokens';
2
- import Dec from 'decimal.js';
3
- import { Client } from 'viem';
4
- import {
5
- createViemContractFromConfigFunc,
6
- SparkIncentiveDataProviderContractViem,
7
- SparkRewardsControllerViem,
8
- } from '../contracts';
9
- import { EthAddress, NetworkNumber } from '../types/common';
10
- import { ClaimType, SparkAirdropType, SparkRewardsClaimableToken } from '../types/claiming';
11
- import { compareAddresses, getEthAmountForDecimals } from '../services/utils';
12
-
13
- const IGNITION_REWARDS = '0xCBA0C0a2a0B6Bb11233ec4EA85C5bFfea33e724d';
14
- const PFL3_REWARDS = '0x7ac96180C4d6b2A328D3a19ac059D0E7Fc3C6d41';
15
-
16
- export const fetchSpkAirdropRewards = async (walletAddress: EthAddress) => {
17
- try {
18
- const res = await fetch(`https://fe.defisaver.com/api/spark/get-spk-airdrop-rewards/${walletAddress}`,
19
- { signal: AbortSignal.timeout(3000) });
20
-
21
- if (!res.ok) throw new Error(await res.text());
22
-
23
- return await res.json();
24
- } catch (err) {
25
- console.error('External API Error: Error fetching SPK airdrop rewards:', err);
26
- return [];
27
- }
28
- };
29
-
30
- type CumulativeClaimedSpkAirdropFuncProp = {
31
- user: string, tokenAddress: string, epoch: number, type: SparkAirdropType,
32
- };
33
-
34
- const getCumulativeClaimedSpkAirdrop = async (provider: Client, network: NetworkNumber, data: CumulativeClaimedSpkAirdropFuncProp[]) => {
35
- const contractIgnition = createViemContractFromConfigFunc('SparkAirdrop', IGNITION_REWARDS)(provider, network);
36
- const contractPFL = createViemContractFromConfigFunc('SparkAirdrop', PFL3_REWARDS)(provider, network);
37
-
38
- const cumulativeAmounts = await Promise.all(data.map((airdrop) => {
39
- const {
40
- user, tokenAddress, epoch, type,
41
- } = airdrop;
42
- const contract = type === SparkAirdropType.SPARK_IGNITION ? contractIgnition : contractPFL;
43
- return contract.read.cumulativeClaimed([user as EthAddress, tokenAddress as EthAddress, BigInt(epoch)]);
44
- }));
45
-
46
- return cumulativeAmounts.map((amount: bigint, i: number) => assetAmountInEth(amount.toString(), getAssetInfoByAddress(data[i].tokenAddress).symbol).toString());
47
- };
48
-
49
- export const fetchSparkAirdropRewards = async (
50
- provider: Client,
51
- network: NetworkNumber,
52
- walletAddresses: EthAddress[],
53
- ) => {
54
- // Fetch all API data in parallel (these are external API calls, can't be batched with multicall)
55
- const apiDataPromises = walletAddresses.map(address => fetchSpkAirdropRewards(address));
56
- const apiDataArray = await Promise.all(apiDataPromises);
57
-
58
- // Process API data and collect contract call data
59
- const allClaimData: Array<{ walletAddress: EthAddress; data: any[] }> = [];
60
- const contractCallsData: CumulativeClaimedSpkAirdropFuncProp[] = [];
61
-
62
- for (let i = 0; i < walletAddresses.length; i++) {
63
- const walletAddress = walletAddresses[i];
64
- const data = apiDataArray[i];
65
-
66
- const falseResponse: boolean = !data;
67
- const emptyResponse: boolean = data && data.length === 0;
68
- const singleResponseWithoutAmount: boolean = data && data.length === 1 && !+data[0].amount;
69
-
70
- if (falseResponse || emptyResponse || singleResponseWithoutAmount) {
71
- allClaimData.push({ walletAddress, data: [] });
72
- continue;
73
- }
74
-
75
- // Filter out IGNITION_REWARDS since they are no longer active
76
- const filteredData = data.filter((rewardInfo: { type: SparkAirdropType; }) => rewardInfo.type !== SparkAirdropType.SPARK_IGNITION);
77
-
78
- allClaimData.push({ walletAddress, data: filteredData });
79
-
80
- // Collect contract call data
81
- filteredData.forEach((airdropInfo: { wallet_address: any; token_address: any; epoch: any; type: any; }) => {
82
- contractCallsData.push({
83
- user: airdropInfo.wallet_address,
84
- tokenAddress: airdropInfo.token_address,
85
- epoch: airdropInfo.epoch,
86
- type: airdropInfo.type,
87
- });
88
- });
89
- }
90
-
91
- // Batch all contract calls
92
- const cumulativeClaimedAmounts = await getCumulativeClaimedSpkAirdrop(provider, network, contractCallsData);
93
-
94
- // Process results
95
- const results: Record<string, any[]> = {};
96
- let contractCallIndex = 0;
97
-
98
- for (const { walletAddress, data } of allClaimData) {
99
- if (data.length === 0) {
100
- results[walletAddress.toLowerCase() as EthAddress] = [];
101
- continue;
102
- }
103
-
104
- const processedRewards = [];
105
- for (let i = 0; i < data.length; i++) {
106
- const rewardInfo = data[i];
107
- const assetInfo = getAssetInfoByAddress(rewardInfo.token_address);
108
- const claimedAmount = cumulativeClaimedAmounts[contractCallIndex];
109
- contractCallIndex++;
110
-
111
- const amount = new Dec(rewardInfo.amount_normalized).minus(claimedAmount).toString();
112
-
113
- if (new Dec(amount).gt('0')) {
114
- processedRewards.push({
115
- symbol: assetInfo.symbol,
116
- underlyingSymbol: assetInfo.symbol,
117
- amount,
118
- claimType: ClaimType.SPARK_AIRDROP,
119
- tokenAddress: rewardInfo.token_address,
120
- walletAddress: rewardInfo.wallet_address,
121
- label: 'Spark Airdrop',
122
- additionalClaimFields: {
123
- merkleRoot: rewardInfo.root_hash,
124
- merkleProofs: rewardInfo.proof,
125
- epoch: rewardInfo.epoch,
126
- rewardType: rewardInfo.type,
127
- allRewardsAmount: rewardInfo.amount_normalized,
128
- },
129
- });
130
- }
131
- }
132
-
133
- results[walletAddress.toLowerCase() as EthAddress] = processedRewards;
134
- }
135
-
136
- return results;
137
- };
138
-
139
-
140
- export const fetchSparkRewards = async (
141
- provider: Client,
142
- network: NetworkNumber,
143
- walletAddress: EthAddress,
144
- marketAddress: EthAddress,
145
- ): Promise<SparkRewardsClaimableToken[]> => {
146
- const contract = SparkIncentiveDataProviderContractViem(provider, network);
147
- const tokensData = await contract.read.getUserReservesIncentivesData([marketAddress, walletAddress]);
148
- const allTokensDataArrays = tokensData.reduce((acc: any[], val) => {
149
- acc.push(val.aTokenIncentivesUserData.userRewardsInformation);
150
- acc.push(val.vTokenIncentivesUserData.userRewardsInformation);
151
- return acc;
152
- }, []);
153
- const allATokens = tokensData.reduce((acc, val) => {
154
- acc.push(val.aTokenIncentivesUserData.tokenAddress);
155
- acc.push(val.vTokenIncentivesUserData.tokenAddress);
156
- return acc;
157
- }, [] as EthAddress[]);
158
-
159
- type RewardsPerAAsset = {
160
- amount: string,
161
- symbol: string,
162
- underlyingAsset: string,
163
- rewardTokenAddress: EthAddress,
164
- aTokenAddress: string,
165
- };
166
- const rewardsPerAAsset = await Promise.all(allATokens.map((token: EthAddress) => {
167
- const sparkRewardsContract = SparkRewardsControllerViem(provider, network);
168
- return sparkRewardsContract.read.getAllUserRewards([[token], walletAddress]);
169
- }));
170
-
171
- // match amounts to token symbol, parse decimal amounts
172
- const rewardsPerAAssetWithInfo = rewardsPerAAsset.map((rewardForAAsset, aaIndex) => {
173
- const rewardsList = rewardForAAsset[0];
174
- const amounts = rewardForAAsset[1];
175
- const rewardsDataArraysForAAsset = allTokensDataArrays[aaIndex];
176
- const aTokenAddress = allATokens[aaIndex];
177
- return rewardsList.map((rewardTokenAddress: EthAddress, rewardIndex: number) => {
178
- const dataArrIndex = rewardsDataArraysForAAsset.findIndex((arr: { rewardTokenAddress: string | undefined; }) => compareAddresses(arr.rewardTokenAddress, rewardTokenAddress));
179
- const amountWei = amounts[rewardIndex];
180
- const amount = getEthAmountForDecimals(amountWei.toString(), rewardsDataArraysForAAsset[dataArrIndex]?.rewardTokenDecimals);
181
- const symbol = rewardsDataArraysForAAsset[dataArrIndex]?.rewardTokenSymbol || '';
182
- const underlyingAsset = symbol;
183
- return ({
184
- amount,
185
- symbol,
186
- underlyingAsset,
187
- rewardTokenAddress,
188
- aTokenAddress,
189
- });
190
- }) as RewardsPerAAsset[];
191
- });
192
-
193
- // sum all unclaimed rewards of all markets per each token
194
- // (e.g. how much awstETH is claimable in total from both aUSDC and awstETH markets)
195
- const totalUnclaimedPerRewardToken: Record<string, any> = {};
196
- rewardsPerAAssetWithInfo
197
- .flat()
198
- .forEach(({
199
- amount, symbol, underlyingAsset, rewardTokenAddress, aTokenAddress,
200
- }) => {
201
- if (+amount > 0) {
202
- if (!totalUnclaimedPerRewardToken[symbol]) {
203
- totalUnclaimedPerRewardToken[symbol] = {
204
- amount, symbol, underlyingAsset, rewardTokenAddress, aTokenAddresses: [aTokenAddress],
205
- };
206
- } else {
207
- totalUnclaimedPerRewardToken[symbol].amount =
208
- new Dec(totalUnclaimedPerRewardToken[symbol].amount).add(amount).toString();
209
- totalUnclaimedPerRewardToken[symbol].aTokenAddresses.push(aTokenAddress);
210
- }
211
- }
212
- }, []);
213
-
214
- return Object.values(totalUnclaimedPerRewardToken).map((reward: any) => ({
215
- symbol: reward.symbol,
216
- amount: reward.amount,
217
- claimType: ClaimType.SPARK_REWARDS,
218
- tokenAddress: reward.rewardTokenAddress,
219
- underlyingSymbol: reward.underlyingAsset,
220
- walletAddress,
221
- label: 'Spark',
222
- additionalClaimFields: {
223
- sparkAssetAddresses: reward.aTokenAddresses,
224
- },
225
- }));
1
+ import { assetAmountInEth, getAssetInfoByAddress } from '@defisaver/tokens';
2
+ import Dec from 'decimal.js';
3
+ import { Client } from 'viem';
4
+ import {
5
+ createViemContractFromConfigFunc,
6
+ SparkIncentiveDataProviderContractViem,
7
+ SparkRewardsControllerViem,
8
+ } from '../contracts';
9
+ import { EthAddress, NetworkNumber } from '../types/common';
10
+ import { ClaimType, SparkAirdropType, SparkRewardsClaimableToken } from '../types/claiming';
11
+ import { compareAddresses, getEthAmountForDecimals } from '../services/utils';
12
+
13
+ const IGNITION_REWARDS = '0xCBA0C0a2a0B6Bb11233ec4EA85C5bFfea33e724d';
14
+ const PFL3_REWARDS = '0x7ac96180C4d6b2A328D3a19ac059D0E7Fc3C6d41';
15
+
16
+ export const fetchSpkAirdropRewards = async (walletAddress: EthAddress) => {
17
+ try {
18
+ const res = await fetch(`https://fe.defisaver.com/api/spark/get-spk-airdrop-rewards/${walletAddress}`,
19
+ { signal: AbortSignal.timeout(3000) });
20
+
21
+ if (!res.ok) throw new Error(await res.text());
22
+
23
+ return await res.json();
24
+ } catch (err) {
25
+ console.error('External API Error: Error fetching SPK airdrop rewards:', err);
26
+ return [];
27
+ }
28
+ };
29
+
30
+ type CumulativeClaimedSpkAirdropFuncProp = {
31
+ user: string, tokenAddress: string, epoch: number, type: SparkAirdropType,
32
+ };
33
+
34
+ const getCumulativeClaimedSpkAirdrop = async (provider: Client, network: NetworkNumber, data: CumulativeClaimedSpkAirdropFuncProp[]) => {
35
+ const contractIgnition = createViemContractFromConfigFunc('SparkAirdrop', IGNITION_REWARDS)(provider, network);
36
+ const contractPFL = createViemContractFromConfigFunc('SparkAirdrop', PFL3_REWARDS)(provider, network);
37
+
38
+ const cumulativeAmounts = await Promise.all(data.map((airdrop) => {
39
+ const {
40
+ user, tokenAddress, epoch, type,
41
+ } = airdrop;
42
+ const contract = type === SparkAirdropType.SPARK_IGNITION ? contractIgnition : contractPFL;
43
+ return contract.read.cumulativeClaimed([user as EthAddress, tokenAddress as EthAddress, BigInt(epoch)]);
44
+ }));
45
+
46
+ return cumulativeAmounts.map((amount: bigint, i: number) => assetAmountInEth(amount.toString(), getAssetInfoByAddress(data[i].tokenAddress).symbol).toString());
47
+ };
48
+
49
+ export const fetchSparkAirdropRewards = async (
50
+ provider: Client,
51
+ network: NetworkNumber,
52
+ walletAddresses: EthAddress[],
53
+ ) => {
54
+ // Fetch all API data in parallel (these are external API calls, can't be batched with multicall)
55
+ const apiDataPromises = walletAddresses.map(address => fetchSpkAirdropRewards(address));
56
+ const apiDataArray = await Promise.all(apiDataPromises);
57
+
58
+ // Process API data and collect contract call data
59
+ const allClaimData: Array<{ walletAddress: EthAddress; data: any[] }> = [];
60
+ const contractCallsData: CumulativeClaimedSpkAirdropFuncProp[] = [];
61
+
62
+ for (let i = 0; i < walletAddresses.length; i++) {
63
+ const walletAddress = walletAddresses[i];
64
+ const data = apiDataArray[i];
65
+
66
+ const falseResponse: boolean = !data;
67
+ const emptyResponse: boolean = data && data.length === 0;
68
+ const singleResponseWithoutAmount: boolean = data && data.length === 1 && !+data[0].amount;
69
+
70
+ if (falseResponse || emptyResponse || singleResponseWithoutAmount) {
71
+ allClaimData.push({ walletAddress, data: [] });
72
+ continue;
73
+ }
74
+
75
+ // Filter out IGNITION_REWARDS since they are no longer active
76
+ const filteredData = data.filter((rewardInfo: { type: SparkAirdropType; }) => rewardInfo.type !== SparkAirdropType.SPARK_IGNITION);
77
+
78
+ allClaimData.push({ walletAddress, data: filteredData });
79
+
80
+ // Collect contract call data
81
+ filteredData.forEach((airdropInfo: { wallet_address: any; token_address: any; epoch: any; type: any; }) => {
82
+ contractCallsData.push({
83
+ user: airdropInfo.wallet_address,
84
+ tokenAddress: airdropInfo.token_address,
85
+ epoch: airdropInfo.epoch,
86
+ type: airdropInfo.type,
87
+ });
88
+ });
89
+ }
90
+
91
+ // Batch all contract calls
92
+ const cumulativeClaimedAmounts = await getCumulativeClaimedSpkAirdrop(provider, network, contractCallsData);
93
+
94
+ // Process results
95
+ const results: Record<string, any[]> = {};
96
+ let contractCallIndex = 0;
97
+
98
+ for (const { walletAddress, data } of allClaimData) {
99
+ if (data.length === 0) {
100
+ results[walletAddress.toLowerCase() as EthAddress] = [];
101
+ continue;
102
+ }
103
+
104
+ const processedRewards = [];
105
+ for (let i = 0; i < data.length; i++) {
106
+ const rewardInfo = data[i];
107
+ const assetInfo = getAssetInfoByAddress(rewardInfo.token_address);
108
+ const claimedAmount = cumulativeClaimedAmounts[contractCallIndex];
109
+ contractCallIndex++;
110
+
111
+ const amount = new Dec(rewardInfo.amount_normalized).minus(claimedAmount).toString();
112
+
113
+ if (new Dec(amount).gt('0')) {
114
+ processedRewards.push({
115
+ symbol: assetInfo.symbol,
116
+ underlyingSymbol: assetInfo.symbol,
117
+ amount,
118
+ claimType: ClaimType.SPARK_AIRDROP,
119
+ tokenAddress: rewardInfo.token_address,
120
+ walletAddress: rewardInfo.wallet_address,
121
+ label: 'Spark Airdrop',
122
+ additionalClaimFields: {
123
+ merkleRoot: rewardInfo.root_hash,
124
+ merkleProofs: rewardInfo.proof,
125
+ epoch: rewardInfo.epoch,
126
+ rewardType: rewardInfo.type,
127
+ allRewardsAmount: rewardInfo.amount_normalized,
128
+ },
129
+ });
130
+ }
131
+ }
132
+
133
+ results[walletAddress.toLowerCase() as EthAddress] = processedRewards;
134
+ }
135
+
136
+ return results;
137
+ };
138
+
139
+
140
+ export const fetchSparkRewards = async (
141
+ provider: Client,
142
+ network: NetworkNumber,
143
+ walletAddress: EthAddress,
144
+ marketAddress: EthAddress,
145
+ ): Promise<SparkRewardsClaimableToken[]> => {
146
+ const contract = SparkIncentiveDataProviderContractViem(provider, network);
147
+ const tokensData = await contract.read.getUserReservesIncentivesData([marketAddress, walletAddress]);
148
+ const allTokensDataArrays = tokensData.reduce((acc: any[], val) => {
149
+ acc.push(val.aTokenIncentivesUserData.userRewardsInformation);
150
+ acc.push(val.vTokenIncentivesUserData.userRewardsInformation);
151
+ return acc;
152
+ }, []);
153
+ const allATokens = tokensData.reduce((acc, val) => {
154
+ acc.push(val.aTokenIncentivesUserData.tokenAddress);
155
+ acc.push(val.vTokenIncentivesUserData.tokenAddress);
156
+ return acc;
157
+ }, [] as EthAddress[]);
158
+
159
+ type RewardsPerAAsset = {
160
+ amount: string,
161
+ symbol: string,
162
+ underlyingAsset: string,
163
+ rewardTokenAddress: EthAddress,
164
+ aTokenAddress: string,
165
+ };
166
+ const rewardsPerAAsset = await Promise.all(allATokens.map((token: EthAddress) => {
167
+ const sparkRewardsContract = SparkRewardsControllerViem(provider, network);
168
+ return sparkRewardsContract.read.getAllUserRewards([[token], walletAddress]);
169
+ }));
170
+
171
+ // match amounts to token symbol, parse decimal amounts
172
+ const rewardsPerAAssetWithInfo = rewardsPerAAsset.map((rewardForAAsset, aaIndex) => {
173
+ const rewardsList = rewardForAAsset[0];
174
+ const amounts = rewardForAAsset[1];
175
+ const rewardsDataArraysForAAsset = allTokensDataArrays[aaIndex];
176
+ const aTokenAddress = allATokens[aaIndex];
177
+ return rewardsList.map((rewardTokenAddress: EthAddress, rewardIndex: number) => {
178
+ const dataArrIndex = rewardsDataArraysForAAsset.findIndex((arr: { rewardTokenAddress: string | undefined; }) => compareAddresses(arr.rewardTokenAddress, rewardTokenAddress));
179
+ const amountWei = amounts[rewardIndex];
180
+ const amount = getEthAmountForDecimals(amountWei.toString(), rewardsDataArraysForAAsset[dataArrIndex]?.rewardTokenDecimals);
181
+ const symbol = rewardsDataArraysForAAsset[dataArrIndex]?.rewardTokenSymbol || '';
182
+ const underlyingAsset = symbol;
183
+ return ({
184
+ amount,
185
+ symbol,
186
+ underlyingAsset,
187
+ rewardTokenAddress,
188
+ aTokenAddress,
189
+ });
190
+ }) as RewardsPerAAsset[];
191
+ });
192
+
193
+ // sum all unclaimed rewards of all markets per each token
194
+ // (e.g. how much awstETH is claimable in total from both aUSDC and awstETH markets)
195
+ const totalUnclaimedPerRewardToken: Record<string, any> = {};
196
+ rewardsPerAAssetWithInfo
197
+ .flat()
198
+ .forEach(({
199
+ amount, symbol, underlyingAsset, rewardTokenAddress, aTokenAddress,
200
+ }) => {
201
+ if (+amount > 0) {
202
+ if (!totalUnclaimedPerRewardToken[symbol]) {
203
+ totalUnclaimedPerRewardToken[symbol] = {
204
+ amount, symbol, underlyingAsset, rewardTokenAddress, aTokenAddresses: [aTokenAddress],
205
+ };
206
+ } else {
207
+ totalUnclaimedPerRewardToken[symbol].amount =
208
+ new Dec(totalUnclaimedPerRewardToken[symbol].amount).add(amount).toString();
209
+ totalUnclaimedPerRewardToken[symbol].aTokenAddresses.push(aTokenAddress);
210
+ }
211
+ }
212
+ }, []);
213
+
214
+ return Object.values(totalUnclaimedPerRewardToken).map((reward: any) => ({
215
+ symbol: reward.symbol,
216
+ amount: reward.amount,
217
+ claimType: ClaimType.SPARK_REWARDS,
218
+ tokenAddress: reward.rewardTokenAddress,
219
+ underlyingSymbol: reward.underlyingAsset,
220
+ walletAddress,
221
+ label: 'Spark',
222
+ additionalClaimFields: {
223
+ sparkAssetAddresses: reward.aTokenAddresses,
224
+ },
225
+ }));
226
226
  };