@defisaver/positions-sdk 2.1.29 → 2.1.31-syrup-dev

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 (133) 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/config/contracts.d.ts +57 -0
  6. package/cjs/config/contracts.js +4 -1
  7. package/cjs/contracts.d.ts +46351 -0
  8. package/cjs/contracts.js +20 -2
  9. package/cjs/helpers/morphoBlueHelpers/index.js +66 -66
  10. package/cjs/markets/aave/marketAssets.js +2 -2
  11. package/cjs/savings/index.d.ts +4 -3
  12. package/cjs/savings/index.js +9 -1
  13. package/cjs/savings/morphoVaults/index.js +17 -17
  14. package/cjs/savings/sparkSavingsVaults/index.d.ts +7 -0
  15. package/cjs/savings/sparkSavingsVaults/index.js +98 -0
  16. package/cjs/savings/sparkSavingsVaults/options.d.ts +6 -0
  17. package/cjs/savings/sparkSavingsVaults/options.js +35 -0
  18. package/cjs/types/savings/index.d.ts +3 -1
  19. package/cjs/types/savings/index.js +1 -0
  20. package/cjs/types/savings/sparkSavingsVaults.d.ts +14 -0
  21. package/cjs/types/savings/sparkSavingsVaults.js +9 -0
  22. package/esm/config/contracts.d.ts +57 -0
  23. package/esm/config/contracts.js +3 -0
  24. package/esm/contracts.d.ts +46351 -0
  25. package/esm/contracts.js +16 -0
  26. package/esm/helpers/morphoBlueHelpers/index.js +66 -66
  27. package/esm/markets/aave/marketAssets.js +2 -2
  28. package/esm/savings/index.d.ts +4 -3
  29. package/esm/savings/index.js +8 -1
  30. package/esm/savings/morphoVaults/index.js +17 -17
  31. package/esm/savings/sparkSavingsVaults/index.d.ts +7 -0
  32. package/esm/savings/sparkSavingsVaults/index.js +57 -0
  33. package/esm/savings/sparkSavingsVaults/options.d.ts +6 -0
  34. package/esm/savings/sparkSavingsVaults/options.js +31 -0
  35. package/esm/types/savings/index.d.ts +3 -1
  36. package/esm/types/savings/index.js +1 -0
  37. package/esm/types/savings/sparkSavingsVaults.d.ts +14 -0
  38. package/esm/types/savings/sparkSavingsVaults.js +6 -0
  39. package/package.json +48 -48
  40. package/src/aaveV2/index.ts +240 -240
  41. package/src/aaveV3/index.ts +614 -614
  42. package/src/aaveV3/merit.ts +97 -97
  43. package/src/aaveV3/merkl.ts +74 -74
  44. package/src/claiming/aaveV3.ts +154 -154
  45. package/src/claiming/compV3.ts +22 -22
  46. package/src/claiming/ethena.ts +61 -61
  47. package/src/claiming/index.ts +12 -12
  48. package/src/claiming/king.ts +66 -66
  49. package/src/claiming/morphoBlue.ts +118 -118
  50. package/src/claiming/spark.ts +225 -225
  51. package/src/compoundV2/index.ts +244 -244
  52. package/src/compoundV3/index.ts +274 -274
  53. package/src/config/contracts.ts +1285 -1282
  54. package/src/constants/index.ts +10 -10
  55. package/src/contracts.ts +160 -142
  56. package/src/curveUsd/index.ts +254 -254
  57. package/src/eulerV2/index.ts +324 -324
  58. package/src/exchange/index.ts +25 -25
  59. package/src/fluid/index.ts +1800 -1800
  60. package/src/helpers/aaveHelpers/index.ts +187 -187
  61. package/src/helpers/compoundHelpers/index.ts +283 -283
  62. package/src/helpers/curveUsdHelpers/index.ts +40 -40
  63. package/src/helpers/eulerHelpers/index.ts +222 -222
  64. package/src/helpers/fluidHelpers/index.ts +326 -326
  65. package/src/helpers/index.ts +10 -10
  66. package/src/helpers/liquityV2Helpers/index.ts +82 -82
  67. package/src/helpers/llamaLendHelpers/index.ts +53 -53
  68. package/src/helpers/makerHelpers/index.ts +52 -52
  69. package/src/helpers/morphoBlueHelpers/index.ts +396 -396
  70. package/src/helpers/sparkHelpers/index.ts +158 -158
  71. package/src/index.ts +49 -49
  72. package/src/liquity/index.ts +159 -159
  73. package/src/liquityV2/index.ts +703 -703
  74. package/src/llamaLend/index.ts +305 -305
  75. package/src/maker/index.ts +223 -223
  76. package/src/markets/aave/index.ts +118 -118
  77. package/src/markets/aave/marketAssets.ts +54 -54
  78. package/src/markets/compound/index.ts +243 -243
  79. package/src/markets/compound/marketsAssets.ts +97 -97
  80. package/src/markets/curveUsd/index.ts +69 -69
  81. package/src/markets/euler/index.ts +26 -26
  82. package/src/markets/fluid/index.ts +2900 -2900
  83. package/src/markets/index.ts +25 -25
  84. package/src/markets/liquityV2/index.ts +102 -102
  85. package/src/markets/llamaLend/contractAddresses.ts +141 -141
  86. package/src/markets/llamaLend/index.ts +235 -235
  87. package/src/markets/morphoBlue/index.ts +971 -971
  88. package/src/markets/spark/index.ts +29 -29
  89. package/src/markets/spark/marketAssets.ts +12 -12
  90. package/src/moneymarket/moneymarketCommonService.ts +85 -85
  91. package/src/morphoBlue/index.ts +274 -274
  92. package/src/portfolio/index.ts +598 -598
  93. package/src/savings/index.ts +55 -46
  94. package/src/savings/makerDsr/index.ts +53 -53
  95. package/src/savings/makerDsr/options.ts +9 -9
  96. package/src/savings/morphoVaults/index.ts +80 -80
  97. package/src/savings/morphoVaults/options.ts +203 -203
  98. package/src/savings/sparkSavingsVaults/index.ts +61 -0
  99. package/src/savings/sparkSavingsVaults/options.ts +36 -0
  100. package/src/savings/yearnVaults/index.ts +73 -73
  101. package/src/savings/yearnVaults/options.ts +32 -32
  102. package/src/services/priceService.ts +278 -278
  103. package/src/services/utils.ts +115 -115
  104. package/src/services/viem.ts +34 -34
  105. package/src/setup.ts +8 -8
  106. package/src/spark/index.ts +456 -456
  107. package/src/staking/eligibility.ts +53 -53
  108. package/src/staking/index.ts +1 -1
  109. package/src/staking/staking.ts +186 -186
  110. package/src/types/aave.ts +196 -196
  111. package/src/types/claiming.ts +114 -114
  112. package/src/types/common.ts +107 -107
  113. package/src/types/compound.ts +144 -144
  114. package/src/types/curveUsd.ts +123 -123
  115. package/src/types/euler.ts +175 -175
  116. package/src/types/fluid.ts +483 -483
  117. package/src/types/index.ts +14 -14
  118. package/src/types/liquity.ts +30 -30
  119. package/src/types/liquityV2.ts +126 -126
  120. package/src/types/llamaLend.ts +159 -159
  121. package/src/types/maker.ts +63 -63
  122. package/src/types/merit.ts +1 -1
  123. package/src/types/merkl.ts +70 -70
  124. package/src/types/morphoBlue.ts +200 -200
  125. package/src/types/portfolio.ts +60 -60
  126. package/src/types/savings/index.ts +20 -18
  127. package/src/types/savings/makerDsr.ts +13 -13
  128. package/src/types/savings/morphoVaults.ts +33 -33
  129. package/src/types/savings/sparkSavingsVaults.ts +16 -0
  130. package/src/types/savings/yearnVaults.ts +14 -14
  131. package/src/types/spark.ts +133 -133
  132. package/src/umbrella/index.ts +69 -69
  133. 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
  };