@defisaver/positions-sdk 2.1.52 → 2.1.53

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 (157) hide show
  1. package/.mocharc.json +4 -4
  2. package/.nvmrc +1 -1
  3. package/README.md +64 -64
  4. package/cjs/fluid/index.d.ts +6 -6
  5. package/cjs/helpers/aaveHelpers/index.d.ts +2 -2
  6. package/cjs/helpers/aaveHelpers/index.js +16 -5
  7. package/cjs/helpers/compoundHelpers/index.js +15 -18
  8. package/cjs/helpers/eulerHelpers/index.d.ts +2 -2
  9. package/cjs/helpers/eulerHelpers/index.js +21 -13
  10. package/cjs/helpers/fluidHelpers/index.js +16 -5
  11. package/cjs/helpers/morphoBlueHelpers/index.js +81 -71
  12. package/cjs/helpers/sparkHelpers/index.d.ts +2 -2
  13. package/cjs/helpers/sparkHelpers/index.js +15 -5
  14. package/cjs/moneymarket/moneymarketCommonService.d.ts +3 -3
  15. package/cjs/moneymarket/moneymarketCommonService.js +9 -9
  16. package/cjs/savings/morphoVaults/index.js +17 -17
  17. package/cjs/types/aave.d.ts +3 -3
  18. package/cjs/types/common.d.ts +7 -0
  19. package/cjs/types/common.js +9 -1
  20. package/cjs/types/compound.d.ts +3 -3
  21. package/cjs/types/curveUsd.d.ts +2 -2
  22. package/cjs/types/euler.d.ts +3 -3
  23. package/cjs/types/fluid.d.ts +3 -3
  24. package/cjs/types/index.d.ts +1 -0
  25. package/cjs/types/index.js +1 -0
  26. package/cjs/types/liquityV2.d.ts +3 -3
  27. package/cjs/types/llamaLend.d.ts +2 -2
  28. package/cjs/types/morphoBlue.d.ts +5 -5
  29. package/cjs/types/spark.d.ts +3 -3
  30. package/esm/fluid/index.d.ts +6 -6
  31. package/esm/helpers/aaveHelpers/index.d.ts +2 -2
  32. package/esm/helpers/aaveHelpers/index.js +16 -5
  33. package/esm/helpers/compoundHelpers/index.js +16 -19
  34. package/esm/helpers/eulerHelpers/index.d.ts +2 -2
  35. package/esm/helpers/eulerHelpers/index.js +21 -13
  36. package/esm/helpers/fluidHelpers/index.js +16 -5
  37. package/esm/helpers/morphoBlueHelpers/index.js +82 -72
  38. package/esm/helpers/sparkHelpers/index.d.ts +2 -2
  39. package/esm/helpers/sparkHelpers/index.js +16 -6
  40. package/esm/moneymarket/moneymarketCommonService.d.ts +3 -3
  41. package/esm/moneymarket/moneymarketCommonService.js +9 -9
  42. package/esm/savings/morphoVaults/index.js +17 -17
  43. package/esm/types/aave.d.ts +3 -3
  44. package/esm/types/common.d.ts +7 -0
  45. package/esm/types/common.js +8 -0
  46. package/esm/types/compound.d.ts +3 -3
  47. package/esm/types/curveUsd.d.ts +2 -2
  48. package/esm/types/euler.d.ts +3 -3
  49. package/esm/types/fluid.d.ts +3 -3
  50. package/esm/types/fluid.js +1 -1
  51. package/esm/types/index.d.ts +1 -0
  52. package/esm/types/index.js +1 -0
  53. package/esm/types/liquityV2.d.ts +3 -3
  54. package/esm/types/llamaLend.d.ts +2 -2
  55. package/esm/types/morphoBlue.d.ts +5 -5
  56. package/esm/types/spark.d.ts +3 -3
  57. package/package.json +48 -48
  58. package/src/aaveV2/index.ts +240 -240
  59. package/src/aaveV3/index.ts +635 -635
  60. package/src/aaveV3/merit.ts +97 -97
  61. package/src/aaveV3/merkl.ts +74 -74
  62. package/src/claiming/aaveV3.ts +154 -154
  63. package/src/claiming/compV3.ts +22 -22
  64. package/src/claiming/ethena.ts +61 -61
  65. package/src/claiming/index.ts +12 -12
  66. package/src/claiming/king.ts +66 -66
  67. package/src/claiming/morphoBlue.ts +118 -118
  68. package/src/claiming/spark.ts +225 -225
  69. package/src/compoundV2/index.ts +244 -244
  70. package/src/compoundV3/index.ts +274 -274
  71. package/src/config/contracts.ts +1320 -1320
  72. package/src/constants/index.ts +10 -10
  73. package/src/contracts.ts +171 -171
  74. package/src/curveUsd/index.ts +254 -254
  75. package/src/eulerV2/index.ts +324 -324
  76. package/src/exchange/index.ts +25 -25
  77. package/src/fluid/index.ts +1800 -1800
  78. package/src/helpers/aaveHelpers/index.ts +202 -191
  79. package/src/helpers/compoundHelpers/index.ts +276 -283
  80. package/src/helpers/curveUsdHelpers/index.ts +40 -40
  81. package/src/helpers/eulerHelpers/index.ts +229 -222
  82. package/src/helpers/fluidHelpers/index.ts +335 -326
  83. package/src/helpers/index.ts +10 -10
  84. package/src/helpers/liquityV2Helpers/index.ts +82 -82
  85. package/src/helpers/llamaLendHelpers/index.ts +53 -53
  86. package/src/helpers/makerHelpers/index.ts +52 -52
  87. package/src/helpers/morphoBlueHelpers/index.ts +405 -396
  88. package/src/helpers/sparkHelpers/index.ts +169 -158
  89. package/src/index.ts +49 -49
  90. package/src/liquity/index.ts +159 -159
  91. package/src/liquityV2/index.ts +703 -703
  92. package/src/llamaLend/index.ts +305 -305
  93. package/src/maker/index.ts +223 -223
  94. package/src/markets/aave/index.ts +118 -118
  95. package/src/markets/aave/marketAssets.ts +54 -54
  96. package/src/markets/compound/index.ts +243 -243
  97. package/src/markets/compound/marketsAssets.ts +97 -97
  98. package/src/markets/curveUsd/index.ts +69 -69
  99. package/src/markets/euler/index.ts +26 -26
  100. package/src/markets/fluid/index.ts +2900 -2900
  101. package/src/markets/index.ts +25 -25
  102. package/src/markets/liquityV2/index.ts +102 -102
  103. package/src/markets/llamaLend/contractAddresses.ts +141 -141
  104. package/src/markets/llamaLend/index.ts +235 -235
  105. package/src/markets/morphoBlue/index.ts +988 -988
  106. package/src/markets/spark/index.ts +29 -29
  107. package/src/markets/spark/marketAssets.ts +12 -12
  108. package/src/moneymarket/moneymarketCommonService.ts +84 -85
  109. package/src/morphoBlue/index.ts +274 -274
  110. package/src/portfolio/index.ts +586 -586
  111. package/src/savings/index.ts +95 -95
  112. package/src/savings/makerDsr/index.ts +53 -53
  113. package/src/savings/makerDsr/options.ts +9 -9
  114. package/src/savings/morphoVaults/index.ts +80 -80
  115. package/src/savings/morphoVaults/options.ts +193 -193
  116. package/src/savings/skyOptions/index.ts +95 -95
  117. package/src/savings/skyOptions/options.ts +10 -10
  118. package/src/savings/sparkSavingsVaults/index.ts +60 -60
  119. package/src/savings/sparkSavingsVaults/options.ts +35 -35
  120. package/src/savings/yearnV3Vaults/index.ts +61 -61
  121. package/src/savings/yearnV3Vaults/options.ts +55 -55
  122. package/src/savings/yearnVaults/index.ts +73 -73
  123. package/src/savings/yearnVaults/options.ts +32 -32
  124. package/src/services/priceService.ts +278 -278
  125. package/src/services/utils.ts +115 -115
  126. package/src/services/viem.ts +57 -57
  127. package/src/setup.ts +8 -8
  128. package/src/spark/index.ts +459 -459
  129. package/src/staking/eligibility.ts +53 -53
  130. package/src/staking/index.ts +1 -1
  131. package/src/staking/staking.ts +192 -192
  132. package/src/types/aave.ts +199 -198
  133. package/src/types/claiming.ts +114 -114
  134. package/src/types/common.ts +115 -107
  135. package/src/types/compound.ts +145 -144
  136. package/src/types/curveUsd.ts +123 -123
  137. package/src/types/euler.ts +176 -175
  138. package/src/types/fluid.ts +485 -483
  139. package/src/types/index.ts +16 -15
  140. package/src/types/liquity.ts +30 -30
  141. package/src/types/liquityV2.ts +128 -126
  142. package/src/types/llamaLend.ts +161 -159
  143. package/src/types/maker.ts +63 -63
  144. package/src/types/merit.ts +1 -1
  145. package/src/types/merkl.ts +70 -70
  146. package/src/types/morphoBlue.ts +202 -202
  147. package/src/types/portfolio.ts +60 -60
  148. package/src/types/savings/index.ts +23 -23
  149. package/src/types/savings/makerDsr.ts +13 -13
  150. package/src/types/savings/morphoVaults.ts +32 -32
  151. package/src/types/savings/sky.ts +14 -14
  152. package/src/types/savings/sparkSavingsVaults.ts +15 -15
  153. package/src/types/savings/yearnV3Vaults.ts +17 -17
  154. package/src/types/savings/yearnVaults.ts +14 -14
  155. package/src/types/spark.ts +135 -134
  156. package/src/umbrella/index.ts +69 -69
  157. package/src/umbrella/umbrellaUtils.ts +29 -29
@@ -1,23 +1,23 @@
1
- import { Client } from 'viem';
2
- import { assetAmountInEth, getAssetInfoByAddress } from '@defisaver/tokens';
3
- import { EthAddress, NetworkNumber } from '../types/common';
4
- import { CompV3ViewContractViem } from '../contracts';
5
- import { ClaimType } from '../types/claiming';
6
-
7
- export const getCompoundV3Rewards = async (provider: Client, network: NetworkNumber, user: EthAddress, market: any) => {
8
- const compV3View = CompV3ViewContractViem(provider, network);
9
- const rewards = await compV3View.read.getRewardsOwed([market, user]);
10
- if (!rewards || rewards.owed.toString() === '0' || getAssetInfoByAddress(rewards.token, network).symbol !== 'COMP') return [];
11
- return [{
12
- symbol: 'COMP',
13
- underlyingSymbol: 'COMP',
14
- tokenAddress: rewards.token,
15
- amount: assetAmountInEth(rewards.owed.toString() || 0, 'COMP'),
16
- walletAddress: user,
17
- label: 'Compound V3',
18
- claimType: ClaimType.COMPOUND_V3_COMP,
19
- additionalClaimFields: {
20
- marketAddress: market,
21
- },
22
- }];
1
+ import { Client } from 'viem';
2
+ import { assetAmountInEth, getAssetInfoByAddress } from '@defisaver/tokens';
3
+ import { EthAddress, NetworkNumber } from '../types/common';
4
+ import { CompV3ViewContractViem } from '../contracts';
5
+ import { ClaimType } from '../types/claiming';
6
+
7
+ export const getCompoundV3Rewards = async (provider: Client, network: NetworkNumber, user: EthAddress, market: any) => {
8
+ const compV3View = CompV3ViewContractViem(provider, network);
9
+ const rewards = await compV3View.read.getRewardsOwed([market, user]);
10
+ if (!rewards || rewards.owed.toString() === '0' || getAssetInfoByAddress(rewards.token, network).symbol !== 'COMP') return [];
11
+ return [{
12
+ symbol: 'COMP',
13
+ underlyingSymbol: 'COMP',
14
+ tokenAddress: rewards.token,
15
+ amount: assetAmountInEth(rewards.owed.toString() || 0, 'COMP'),
16
+ walletAddress: user,
17
+ label: 'Compound V3',
18
+ claimType: ClaimType.COMPOUND_V3_COMP,
19
+ additionalClaimFields: {
20
+ marketAddress: market,
21
+ },
22
+ }];
23
23
  };
@@ -1,61 +1,61 @@
1
- import { getAssetInfo } from '@defisaver/tokens';
2
- import Dec from 'decimal.js';
3
- import { getAddress } from 'viem';
4
- import { EthAddress } from '../types/common';
5
- import { ClaimType } from '../types/claiming';
6
- import { getEthAmountForDecimals } from '../services/utils';
7
-
8
- export const fetchEthenaAirdropReward = async (address: EthAddress) => {
9
- try {
10
- const checksumAddress = getAddress(address);
11
- const response = await fetch(`https://airdrop-data-ethena-s4.s3.us-west-2.amazonaws.com/${checksumAddress}/0x3d99219fbd49ace3f48d6ca1340e505ec1bdf27d1f8d0e15ec9f286cc9215fcd-${checksumAddress}.json`);
12
-
13
- if (!response.ok) {
14
- if (response.status === 403) {
15
- // This is also okay, means that there are no rewards for the address
16
- return;
17
- }
18
- throw new Error(`HTTP error! status: ${response.status}`);
19
- }
20
-
21
- return await response.json();
22
- } catch (error) {
23
- console.error('Error fetching Ethena airdrop rewards:', error);
24
- }
25
- };
26
-
27
- export const fetchEthenaAirdropRewards = async (walletAddresses: EthAddress[]): Promise<Record<string, any[]>> => {
28
- const apiDataPromises = walletAddresses.map(address => fetchEthenaAirdropReward(address));
29
- const apiDataArray = await Promise.all(apiDataPromises);
30
-
31
- const results: Record<string, any[]> = {};
32
- for (let i = 0; i < walletAddresses.length; i++) {
33
- const walletAddress = walletAddresses[i];
34
- const data = apiDataArray[i];
35
-
36
- if (!data || data.claimed) {
37
- continue;
38
- }
39
-
40
- const processedRewards = [];
41
- const assetInfo = getAssetInfo('sENA');
42
-
43
- const amount = getEthAmountForDecimals(data.events[0].awardAmount, assetInfo.decimals);
44
-
45
- if (new Dec(amount).gt('0')) {
46
- processedRewards.push({
47
- symbol: assetInfo.symbol,
48
- underlyingSymbol: assetInfo.symbol,
49
- amount,
50
- claimType: ClaimType.ETHENA_AIRDROP,
51
- tokenAddress: assetInfo.address,
52
- walletAddress,
53
- label: 'Ethena Airdrop',
54
- });
55
- }
56
-
57
- results[walletAddress.toLowerCase() as EthAddress] = processedRewards;
58
- }
59
-
60
- return results;
61
- };
1
+ import { getAssetInfo } from '@defisaver/tokens';
2
+ import Dec from 'decimal.js';
3
+ import { getAddress } from 'viem';
4
+ import { EthAddress } from '../types/common';
5
+ import { ClaimType } from '../types/claiming';
6
+ import { getEthAmountForDecimals } from '../services/utils';
7
+
8
+ export const fetchEthenaAirdropReward = async (address: EthAddress) => {
9
+ try {
10
+ const checksumAddress = getAddress(address);
11
+ const response = await fetch(`https://airdrop-data-ethena-s4.s3.us-west-2.amazonaws.com/${checksumAddress}/0x3d99219fbd49ace3f48d6ca1340e505ec1bdf27d1f8d0e15ec9f286cc9215fcd-${checksumAddress}.json`);
12
+
13
+ if (!response.ok) {
14
+ if (response.status === 403) {
15
+ // This is also okay, means that there are no rewards for the address
16
+ return;
17
+ }
18
+ throw new Error(`HTTP error! status: ${response.status}`);
19
+ }
20
+
21
+ return await response.json();
22
+ } catch (error) {
23
+ console.error('Error fetching Ethena airdrop rewards:', error);
24
+ }
25
+ };
26
+
27
+ export const fetchEthenaAirdropRewards = async (walletAddresses: EthAddress[]): Promise<Record<string, any[]>> => {
28
+ const apiDataPromises = walletAddresses.map(address => fetchEthenaAirdropReward(address));
29
+ const apiDataArray = await Promise.all(apiDataPromises);
30
+
31
+ const results: Record<string, any[]> = {};
32
+ for (let i = 0; i < walletAddresses.length; i++) {
33
+ const walletAddress = walletAddresses[i];
34
+ const data = apiDataArray[i];
35
+
36
+ if (!data || data.claimed) {
37
+ continue;
38
+ }
39
+
40
+ const processedRewards = [];
41
+ const assetInfo = getAssetInfo('sENA');
42
+
43
+ const amount = getEthAmountForDecimals(data.events[0].awardAmount, assetInfo.decimals);
44
+
45
+ if (new Dec(amount).gt('0')) {
46
+ processedRewards.push({
47
+ symbol: assetInfo.symbol,
48
+ underlyingSymbol: assetInfo.symbol,
49
+ amount,
50
+ claimType: ClaimType.ETHENA_AIRDROP,
51
+ tokenAddress: assetInfo.address,
52
+ walletAddress,
53
+ label: 'Ethena Airdrop',
54
+ });
55
+ }
56
+
57
+ results[walletAddress.toLowerCase() as EthAddress] = processedRewards;
58
+ }
59
+
60
+ return results;
61
+ };
@@ -1,13 +1,13 @@
1
- import * as aaveV3Claim from './aaveV3';
2
- import * as compV3Claim from './compV3';
3
- import * as kingV3Claim from './king';
4
- import * as morphoBlueClaim from './morphoBlue';
5
- import * as sparkClaim from './spark';
6
-
7
- export {
8
- aaveV3Claim,
9
- compV3Claim,
10
- kingV3Claim,
11
- morphoBlueClaim,
12
- sparkClaim,
1
+ import * as aaveV3Claim from './aaveV3';
2
+ import * as compV3Claim from './compV3';
3
+ import * as kingV3Claim from './king';
4
+ import * as morphoBlueClaim from './morphoBlue';
5
+ import * as sparkClaim from './spark';
6
+
7
+ export {
8
+ aaveV3Claim,
9
+ compV3Claim,
10
+ kingV3Claim,
11
+ morphoBlueClaim,
12
+ sparkClaim,
13
13
  };
@@ -1,66 +1,66 @@
1
- import Dec from 'decimal.js';
2
- import { assetAmountInEth } from '@defisaver/tokens';
3
- import { Client } from 'viem';
4
- import { UUPSViem } from '../contracts';
5
- import { EthAddress, HexString, NetworkNumber } from '../types/common';
6
- import { ClaimType } from '../types/claiming';
7
-
8
- export const fetchKingRewards = async (walletAddress: EthAddress) => {
9
- try {
10
- const res = await fetch(`https://fe.defisaver.com/api/etherfi/get-king-rewards/${walletAddress}`,
11
- { signal: AbortSignal.timeout(5000) });
12
-
13
- if (!res.ok) throw new Error(await res.text());
14
-
15
- return await res.json();
16
- } catch (err) {
17
- console.error('External API Error: Error fetching KING rewards:', err);
18
- return { Amount: '0', Root: '', Proofs: [] };
19
- }
20
- };
21
-
22
- export const getKingRewards = async (provider: Client, network: NetworkNumber, walletAddresses: EthAddress[]) => {
23
- // Fetch all API data in parallel (these are external API calls, can't be batched with multicall)
24
- const apiDataPromises = walletAddresses.map(address => fetchKingRewards(address));
25
- const apiDataArray = await Promise.all(apiDataPromises);
26
-
27
- // Batch all contract calls using multicall
28
- const contract = UUPSViem(provider, network);
29
- const cumulativePromises = walletAddresses.map(address => contract.read.cumulativeClaimed([address]),
30
- );
31
- const cumulativeResults = await Promise.all(cumulativePromises);
32
-
33
- // Process results
34
- const results: Record<string, any[]> = {};
35
-
36
- for (let i = 0; i < walletAddresses.length; i++) {
37
- const walletAddress = walletAddresses[i];
38
- const data = apiDataArray[i];
39
- const cumulative = cumulativeResults[i];
40
-
41
- const allRewardsAmount = assetAmountInEth(data.Amount, 'KING');
42
- const claimedAmount = assetAmountInEth(cumulative.toString(), 'KING');
43
- const amountToClaim = new Dec(allRewardsAmount).sub(claimedAmount);
44
-
45
- if (amountToClaim.lessThanOrEqualTo('0')) {
46
- results[walletAddress.toLowerCase() as EthAddress] = [];
47
- } else {
48
- results[walletAddress.toLowerCase() as EthAddress] = [{
49
- symbol: 'KING',
50
- underlyingSymbol: 'KING',
51
- tokenAddress: '0x8F08B70456eb22f6109F57b8fafE862ED28E6040',
52
- amount: amountToClaim.toString(),
53
- walletAddress,
54
- label: 'weETH',
55
- claimType: ClaimType.KING_REWARDS,
56
- additionalClaimFields: {
57
- allRewardsAmount,
58
- merkleRoot: data.Root,
59
- merkleProofs: data.Proofs,
60
- },
61
- }];
62
- }
63
- }
64
-
65
- return results;
66
- };
1
+ import Dec from 'decimal.js';
2
+ import { assetAmountInEth } from '@defisaver/tokens';
3
+ import { Client } from 'viem';
4
+ import { UUPSViem } from '../contracts';
5
+ import { EthAddress, HexString, NetworkNumber } from '../types/common';
6
+ import { ClaimType } from '../types/claiming';
7
+
8
+ export const fetchKingRewards = async (walletAddress: EthAddress) => {
9
+ try {
10
+ const res = await fetch(`https://fe.defisaver.com/api/etherfi/get-king-rewards/${walletAddress}`,
11
+ { signal: AbortSignal.timeout(5000) });
12
+
13
+ if (!res.ok) throw new Error(await res.text());
14
+
15
+ return await res.json();
16
+ } catch (err) {
17
+ console.error('External API Error: Error fetching KING rewards:', err);
18
+ return { Amount: '0', Root: '', Proofs: [] };
19
+ }
20
+ };
21
+
22
+ export const getKingRewards = async (provider: Client, network: NetworkNumber, walletAddresses: EthAddress[]) => {
23
+ // Fetch all API data in parallel (these are external API calls, can't be batched with multicall)
24
+ const apiDataPromises = walletAddresses.map(address => fetchKingRewards(address));
25
+ const apiDataArray = await Promise.all(apiDataPromises);
26
+
27
+ // Batch all contract calls using multicall
28
+ const contract = UUPSViem(provider, network);
29
+ const cumulativePromises = walletAddresses.map(address => contract.read.cumulativeClaimed([address]),
30
+ );
31
+ const cumulativeResults = await Promise.all(cumulativePromises);
32
+
33
+ // Process results
34
+ const results: Record<string, any[]> = {};
35
+
36
+ for (let i = 0; i < walletAddresses.length; i++) {
37
+ const walletAddress = walletAddresses[i];
38
+ const data = apiDataArray[i];
39
+ const cumulative = cumulativeResults[i];
40
+
41
+ const allRewardsAmount = assetAmountInEth(data.Amount, 'KING');
42
+ const claimedAmount = assetAmountInEth(cumulative.toString(), 'KING');
43
+ const amountToClaim = new Dec(allRewardsAmount).sub(claimedAmount);
44
+
45
+ if (amountToClaim.lessThanOrEqualTo('0')) {
46
+ results[walletAddress.toLowerCase() as EthAddress] = [];
47
+ } else {
48
+ results[walletAddress.toLowerCase() as EthAddress] = [{
49
+ symbol: 'KING',
50
+ underlyingSymbol: 'KING',
51
+ tokenAddress: '0x8F08B70456eb22f6109F57b8fafE862ED28E6040',
52
+ amount: amountToClaim.toString(),
53
+ walletAddress,
54
+ label: 'weETH',
55
+ claimType: ClaimType.KING_REWARDS,
56
+ additionalClaimFields: {
57
+ allRewardsAmount,
58
+ merkleRoot: data.Root,
59
+ merkleProofs: data.Proofs,
60
+ },
61
+ }];
62
+ }
63
+ }
64
+
65
+ return results;
66
+ };
@@ -1,119 +1,119 @@
1
- import { assetAmountInEth, getAssetInfoByAddress } from '@defisaver/tokens';
2
- import { Client } from 'viem';
3
- import Dec from 'decimal.js';
4
- import { ClaimableToken, ClaimType } from '../types/claiming';
5
- import { EthAddress, NetworkNumber } from '../types/common';
6
- import { createViemContractFromConfigFunc } from '../contracts';
7
- import { getMorphoUnderlyingSymbol } from '../helpers/morphoBlueHelpers';
8
-
9
- const MORPHO_ALLOWED_CONTRACTS = [
10
- '0x330eefa8a787552DC5cAd3C3cA644844B1E61Ddb',
11
- '0x678dDC1d07eaa166521325394cDEb1E4c086DF43',
12
- '0x2efd4625d0c149ebadf118ec5446c6de24d916a4',
13
- ];
14
-
15
- const MORPHO_ALLOWED_TOKENS = [
16
- 'MORPHO',
17
- 'MORPHO Legacy',
18
- ];
19
-
20
-
21
- export const getMorphoBlueRewardsInfo = async (address: EthAddress) => {
22
- if (!address) return { claimable: '0' };
23
-
24
- try {
25
- const res = await fetch(`https://rewards.morpho.org/v1/users/${address}/distributions`,
26
- { signal: AbortSignal.timeout(3000) });
27
-
28
- if (!res.ok) throw new Error(await res.text());
29
- return await res.json();
30
- } catch (error) {
31
- console.error('External API Failure: Morpho Merit', error);
32
- return { claimable: '0' };
33
- }
34
- };
35
-
36
- export const fetchMorphoBlueRewards = async (
37
- provider: Client,
38
- network: NetworkNumber,
39
- walletAddresses: EthAddress[],
40
- ): Promise<Record<string, ClaimableToken[]>> => {
41
- // Fetch all API data in parallel (these are external API calls, can't be batched with multicall)
42
- const apiDataPromises = walletAddresses.map(address => getMorphoBlueRewardsInfo(address));
43
- const apiDataArray = await Promise.all(apiDataPromises);
44
-
45
- // Process API data to get claimable tokens for each wallet
46
- const allClaimableTokens: Array<{ walletAddress: EthAddress; tokens: ClaimableToken[] }> = [];
47
- const contractCallsData: Array<{ walletAddress: EthAddress; tokenAddress: EthAddress; distributor: EthAddress }> = [];
48
-
49
- for (let i = 0; i < walletAddresses.length; i++) {
50
- const walletAddress = walletAddresses[i];
51
- const data = apiDataArray[i];
52
-
53
- const claimableTokens = data?.data?.reduce((acc: ClaimableToken[], reward: any) => {
54
- const token = getAssetInfoByAddress(reward.asset.address);
55
- if (!MORPHO_ALLOWED_CONTRACTS.includes(reward?.distributor?.address) || !MORPHO_ALLOWED_TOKENS.includes(token.symbol)) {
56
- return acc;
57
- }
58
-
59
- // Store contract call data for batching
60
- contractCallsData.push({
61
- walletAddress,
62
- tokenAddress: reward.asset.address,
63
- distributor: reward?.distributor?.address,
64
- });
65
-
66
- return [
67
- ...acc,
68
- {
69
- symbol: token.symbol,
70
- underlyingSymbol: getMorphoUnderlyingSymbol(token.symbol),
71
- tokenAddress: reward.asset.address,
72
- amount: assetAmountInEth(reward.claimable, token.symbol),
73
- walletAddress,
74
- label: token.symbol,
75
- claimType: ClaimType.MORPHO,
76
- additionalClaimFields: {
77
- originalAmount: reward.claimable,
78
- distributor: reward?.distributor?.address,
79
- merkleProofs: reward?.proof,
80
- isLegacy: token.symbol === 'MORPHO Legacy',
81
- txData: reward?.tx_data,
82
- },
83
- }];
84
- }, []) || [];
85
-
86
- allClaimableTokens.push({ walletAddress, tokens: claimableTokens });
87
- }
88
-
89
- // Batch all contract calls using multicall
90
- const contractPromises = contractCallsData.map(({ walletAddress, tokenAddress, distributor }) => {
91
- const distributorContract = createViemContractFromConfigFunc('MorphoDistributor', distributor)(provider, network);
92
- return distributorContract.read.claimed([walletAddress, tokenAddress]);
93
- });
94
- const contractResults = await Promise.all(contractPromises);
95
-
96
- // Process results
97
- const results: Record<string, ClaimableToken[]> = {};
98
- let contractCallIndex = 0;
99
-
100
- for (const { walletAddress, tokens } of allClaimableTokens) {
101
- const updatedTokens: ClaimableToken[] = [];
102
-
103
- for (let i = 0; i < tokens.length; i++) {
104
- const claimableToken = tokens[i];
105
- const claimed = assetAmountInEth(contractResults[contractCallIndex].toString(), claimableToken.underlyingSymbol);
106
- contractCallIndex++;
107
-
108
- const updatedToken = { ...claimableToken };
109
- updatedToken.amount = new Dec(claimableToken.amount).sub(claimed).toString();
110
- updatedTokens.push(updatedToken);
111
- }
112
-
113
- results[walletAddress.toLowerCase() as EthAddress] = updatedTokens.filter(
114
- (claimableToken) => new Dec(claimableToken.amount).gt(0),
115
- );
116
- }
117
-
118
- return results;
1
+ import { assetAmountInEth, getAssetInfoByAddress } from '@defisaver/tokens';
2
+ import { Client } from 'viem';
3
+ import Dec from 'decimal.js';
4
+ import { ClaimableToken, ClaimType } from '../types/claiming';
5
+ import { EthAddress, NetworkNumber } from '../types/common';
6
+ import { createViemContractFromConfigFunc } from '../contracts';
7
+ import { getMorphoUnderlyingSymbol } from '../helpers/morphoBlueHelpers';
8
+
9
+ const MORPHO_ALLOWED_CONTRACTS = [
10
+ '0x330eefa8a787552DC5cAd3C3cA644844B1E61Ddb',
11
+ '0x678dDC1d07eaa166521325394cDEb1E4c086DF43',
12
+ '0x2efd4625d0c149ebadf118ec5446c6de24d916a4',
13
+ ];
14
+
15
+ const MORPHO_ALLOWED_TOKENS = [
16
+ 'MORPHO',
17
+ 'MORPHO Legacy',
18
+ ];
19
+
20
+
21
+ export const getMorphoBlueRewardsInfo = async (address: EthAddress) => {
22
+ if (!address) return { claimable: '0' };
23
+
24
+ try {
25
+ const res = await fetch(`https://rewards.morpho.org/v1/users/${address}/distributions`,
26
+ { signal: AbortSignal.timeout(3000) });
27
+
28
+ if (!res.ok) throw new Error(await res.text());
29
+ return await res.json();
30
+ } catch (error) {
31
+ console.error('External API Failure: Morpho Merit', error);
32
+ return { claimable: '0' };
33
+ }
34
+ };
35
+
36
+ export const fetchMorphoBlueRewards = async (
37
+ provider: Client,
38
+ network: NetworkNumber,
39
+ walletAddresses: EthAddress[],
40
+ ): Promise<Record<string, ClaimableToken[]>> => {
41
+ // Fetch all API data in parallel (these are external API calls, can't be batched with multicall)
42
+ const apiDataPromises = walletAddresses.map(address => getMorphoBlueRewardsInfo(address));
43
+ const apiDataArray = await Promise.all(apiDataPromises);
44
+
45
+ // Process API data to get claimable tokens for each wallet
46
+ const allClaimableTokens: Array<{ walletAddress: EthAddress; tokens: ClaimableToken[] }> = [];
47
+ const contractCallsData: Array<{ walletAddress: EthAddress; tokenAddress: EthAddress; distributor: EthAddress }> = [];
48
+
49
+ for (let i = 0; i < walletAddresses.length; i++) {
50
+ const walletAddress = walletAddresses[i];
51
+ const data = apiDataArray[i];
52
+
53
+ const claimableTokens = data?.data?.reduce((acc: ClaimableToken[], reward: any) => {
54
+ const token = getAssetInfoByAddress(reward.asset.address);
55
+ if (!MORPHO_ALLOWED_CONTRACTS.includes(reward?.distributor?.address) || !MORPHO_ALLOWED_TOKENS.includes(token.symbol)) {
56
+ return acc;
57
+ }
58
+
59
+ // Store contract call data for batching
60
+ contractCallsData.push({
61
+ walletAddress,
62
+ tokenAddress: reward.asset.address,
63
+ distributor: reward?.distributor?.address,
64
+ });
65
+
66
+ return [
67
+ ...acc,
68
+ {
69
+ symbol: token.symbol,
70
+ underlyingSymbol: getMorphoUnderlyingSymbol(token.symbol),
71
+ tokenAddress: reward.asset.address,
72
+ amount: assetAmountInEth(reward.claimable, token.symbol),
73
+ walletAddress,
74
+ label: token.symbol,
75
+ claimType: ClaimType.MORPHO,
76
+ additionalClaimFields: {
77
+ originalAmount: reward.claimable,
78
+ distributor: reward?.distributor?.address,
79
+ merkleProofs: reward?.proof,
80
+ isLegacy: token.symbol === 'MORPHO Legacy',
81
+ txData: reward?.tx_data,
82
+ },
83
+ }];
84
+ }, []) || [];
85
+
86
+ allClaimableTokens.push({ walletAddress, tokens: claimableTokens });
87
+ }
88
+
89
+ // Batch all contract calls using multicall
90
+ const contractPromises = contractCallsData.map(({ walletAddress, tokenAddress, distributor }) => {
91
+ const distributorContract = createViemContractFromConfigFunc('MorphoDistributor', distributor)(provider, network);
92
+ return distributorContract.read.claimed([walletAddress, tokenAddress]);
93
+ });
94
+ const contractResults = await Promise.all(contractPromises);
95
+
96
+ // Process results
97
+ const results: Record<string, ClaimableToken[]> = {};
98
+ let contractCallIndex = 0;
99
+
100
+ for (const { walletAddress, tokens } of allClaimableTokens) {
101
+ const updatedTokens: ClaimableToken[] = [];
102
+
103
+ for (let i = 0; i < tokens.length; i++) {
104
+ const claimableToken = tokens[i];
105
+ const claimed = assetAmountInEth(contractResults[contractCallIndex].toString(), claimableToken.underlyingSymbol);
106
+ contractCallIndex++;
107
+
108
+ const updatedToken = { ...claimableToken };
109
+ updatedToken.amount = new Dec(claimableToken.amount).sub(claimed).toString();
110
+ updatedTokens.push(updatedToken);
111
+ }
112
+
113
+ results[walletAddress.toLowerCase() as EthAddress] = updatedTokens.filter(
114
+ (claimableToken) => new Dec(claimableToken.amount).gt(0),
115
+ );
116
+ }
117
+
118
+ return results;
119
119
  };