@defisaver/positions-sdk 2.1.4 → 2.1.5

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 (90) 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/curveUsd/index.js +2 -1
  6. package/cjs/helpers/morphoBlueHelpers/index.js +66 -66
  7. package/cjs/types/curveUsd.d.ts +2 -0
  8. package/esm/curveUsd/index.js +2 -1
  9. package/esm/helpers/morphoBlueHelpers/index.js +66 -66
  10. package/esm/types/curveUsd.d.ts +2 -0
  11. package/package.json +47 -47
  12. package/src/aaveV2/index.ts +240 -240
  13. package/src/aaveV3/index.ts +614 -614
  14. package/src/aaveV3/merit.ts +94 -94
  15. package/src/aaveV3/merkl.ts +74 -74
  16. package/src/claiming/aaveV3.ts +154 -154
  17. package/src/claiming/compV3.ts +22 -22
  18. package/src/claiming/index.ts +12 -12
  19. package/src/claiming/king.ts +66 -66
  20. package/src/claiming/morphoBlue.ts +118 -118
  21. package/src/claiming/spark.ts +225 -225
  22. package/src/compoundV2/index.ts +244 -244
  23. package/src/compoundV3/index.ts +274 -274
  24. package/src/config/contracts.ts +1228 -1228
  25. package/src/constants/index.ts +10 -10
  26. package/src/contracts.ts +120 -120
  27. package/src/curveUsd/index.ts +254 -250
  28. package/src/eulerV2/index.ts +324 -324
  29. package/src/exchange/index.ts +25 -25
  30. package/src/fluid/index.ts +1638 -1638
  31. package/src/helpers/aaveHelpers/index.ts +185 -185
  32. package/src/helpers/compoundHelpers/index.ts +283 -283
  33. package/src/helpers/curveUsdHelpers/index.ts +40 -40
  34. package/src/helpers/eulerHelpers/index.ts +222 -222
  35. package/src/helpers/fluidHelpers/index.ts +326 -326
  36. package/src/helpers/index.ts +10 -10
  37. package/src/helpers/liquityV2Helpers/index.ts +82 -82
  38. package/src/helpers/llamaLendHelpers/index.ts +53 -53
  39. package/src/helpers/makerHelpers/index.ts +52 -52
  40. package/src/helpers/morphoBlueHelpers/index.ts +396 -396
  41. package/src/helpers/sparkHelpers/index.ts +155 -155
  42. package/src/index.ts +47 -47
  43. package/src/liquity/index.ts +159 -159
  44. package/src/liquityV2/index.ts +657 -657
  45. package/src/llamaLend/index.ts +305 -305
  46. package/src/maker/index.ts +223 -223
  47. package/src/markets/aave/index.ts +116 -116
  48. package/src/markets/aave/marketAssets.ts +49 -49
  49. package/src/markets/compound/index.ts +227 -227
  50. package/src/markets/compound/marketsAssets.ts +90 -90
  51. package/src/markets/curveUsd/index.ts +69 -69
  52. package/src/markets/euler/index.ts +26 -26
  53. package/src/markets/fluid/index.ts +2456 -2456
  54. package/src/markets/index.ts +25 -25
  55. package/src/markets/liquityV2/index.ts +102 -102
  56. package/src/markets/llamaLend/contractAddresses.ts +141 -141
  57. package/src/markets/llamaLend/index.ts +235 -235
  58. package/src/markets/morphoBlue/index.ts +895 -895
  59. package/src/markets/spark/index.ts +29 -29
  60. package/src/markets/spark/marketAssets.ts +11 -11
  61. package/src/moneymarket/moneymarketCommonService.ts +80 -80
  62. package/src/morphoBlue/index.ts +274 -274
  63. package/src/portfolio/index.ts +570 -570
  64. package/src/services/priceService.ts +159 -159
  65. package/src/services/utils.ts +99 -99
  66. package/src/services/viem.ts +32 -32
  67. package/src/setup.ts +8 -8
  68. package/src/spark/index.ts +445 -445
  69. package/src/staking/eligibility.ts +59 -59
  70. package/src/staking/index.ts +1 -1
  71. package/src/staking/staking.ts +170 -170
  72. package/src/types/aave.ts +189 -189
  73. package/src/types/claiming.ts +109 -109
  74. package/src/types/common.ts +105 -105
  75. package/src/types/compound.ts +136 -136
  76. package/src/types/curveUsd.ts +123 -121
  77. package/src/types/euler.ts +175 -175
  78. package/src/types/fluid.ts +448 -448
  79. package/src/types/index.ts +13 -13
  80. package/src/types/liquity.ts +30 -30
  81. package/src/types/liquityV2.ts +126 -126
  82. package/src/types/llamaLend.ts +159 -159
  83. package/src/types/maker.ts +63 -63
  84. package/src/types/merit.ts +1 -1
  85. package/src/types/merkl.ts +70 -70
  86. package/src/types/morphoBlue.ts +194 -194
  87. package/src/types/portfolio.ts +60 -60
  88. package/src/types/spark.ts +135 -135
  89. package/src/umbrella/index.ts +69 -69
  90. package/src/umbrella/umbrellaUtils.ts +29 -29
@@ -1,159 +1,159 @@
1
- import Dec from 'decimal.js';
2
- import { getAssetInfo } from '@defisaver/tokens';
3
- import { Client, PublicClient } from 'viem';
4
- import {
5
- COMPPriceFeedContractViem,
6
- BTCPriceFeedContractViem,
7
- ETHPriceFeedContractViem,
8
- USDCPriceFeedContractViem,
9
- WstETHPriceFeedContractViem,
10
- WeETHPriceFeedContractViem,
11
- } from '../contracts';
12
- import { EthAddress, NetworkNumber } from '../types/common';
13
- import { getEthAmountForDecimals } from './utils';
14
-
15
- export const getEthPrice = async (client: Client) => {
16
- const contract = ETHPriceFeedContractViem(client, NetworkNumber.Eth);
17
- const price = await contract.read.latestAnswer();
18
- return new Dec(price.toString()).div(1e8).toString();
19
- };
20
-
21
- export const getUSDCPrice = async (client: Client) => {
22
- const contract = USDCPriceFeedContractViem(client, NetworkNumber.Eth);
23
- const price = await contract.read.latestAnswer();
24
- return new Dec(price.toString()).div(1e8).toString();
25
- };
26
-
27
- export const getCompPrice = async (client: Client) => {
28
- const contract = COMPPriceFeedContractViem(client, NetworkNumber.Eth);
29
- const price = await contract.read.latestAnswer();
30
- return new Dec(price.toString()).div(1e8).toString();
31
- };
32
-
33
- export const getWstETHPrice = async (client: Client, network: NetworkNumber = NetworkNumber.Eth) => {
34
- const wstETHFeedContract = WstETHPriceFeedContractViem(client, network);
35
- const ethFeedContract = ETHPriceFeedContractViem(client, network);
36
-
37
- const [ethPriceWei, wstETHRateWei, decimals] = await Promise.all([
38
- ethFeedContract.read.latestAnswer(),
39
- wstETHFeedContract.read.latestRoundData(),
40
- wstETHFeedContract.read.decimals(),
41
- ]);
42
-
43
- const ethPrice = new Dec(ethPriceWei.toString()).div(1e8);
44
-
45
- const wstETHRate = getEthAmountForDecimals(wstETHRateWei[1].toString(), decimals);
46
-
47
- return new Dec(ethPrice).mul(wstETHRate).toString();
48
- };
49
-
50
- export const getWeETHPrice = async (client: Client, network: NetworkNumber = NetworkNumber.Eth) => {
51
- const weETHFeedContract = WeETHPriceFeedContractViem(client, network);
52
- const ethFeedContract = ETHPriceFeedContractViem(client, network);
53
-
54
- const [ethPriceWei, weETHRateWei, decimals] = await Promise.all([
55
- ethFeedContract.read.latestAnswer(),
56
- weETHFeedContract.read.latestRoundData(),
57
- weETHFeedContract.read.decimals(),
58
- ]);
59
-
60
- const ethPrice = new Dec(ethPriceWei.toString()).div(1e8);
61
-
62
- const weETHRate = getEthAmountForDecimals(weETHRateWei[1].toString(), decimals);
63
-
64
- return new Dec(ethPrice).mul(weETHRate).toString();
65
- };
66
-
67
- export const getWstETHChainLinkPriceCalls = (client: PublicClient, network: NetworkNumber) => {
68
- const wstETHFeedContract = WstETHPriceFeedContractViem(client, network);
69
- const ethFeedContract = ETHPriceFeedContractViem(client, network);
70
- const calls = [
71
- {
72
- address: ethFeedContract.address,
73
- abi: ethFeedContract.abi,
74
- functionName: 'latestAnswer',
75
- args: [],
76
- },
77
- {
78
- address: wstETHFeedContract.address,
79
- abi: wstETHFeedContract.abi,
80
- functionName: 'latestRoundData',
81
- args: [],
82
- },
83
- {
84
- address: wstETHFeedContract.address,
85
- abi: wstETHFeedContract.abi,
86
- functionName: 'decimals',
87
- args: [],
88
- },
89
- ];
90
- return calls;
91
- };
92
-
93
- export const getWeETHChainLinkPriceCalls = (client: PublicClient, network: NetworkNumber) => {
94
- const weETHFeedContract = WeETHPriceFeedContractViem(client, network);
95
- const ethFeedContract = ETHPriceFeedContractViem(client, network);
96
- const calls = [
97
- {
98
- address: ethFeedContract.address,
99
- abi: ethFeedContract.abi,
100
- functionName: 'latestAnswer',
101
- args: [],
102
- },
103
- {
104
- address: weETHFeedContract.address,
105
- abi: weETHFeedContract.abi,
106
- functionName: 'latestRoundData',
107
- args: [],
108
- },
109
- {
110
- address: weETHFeedContract.address,
111
- abi: weETHFeedContract.abi,
112
- functionName: 'decimals',
113
- args: [],
114
- },
115
- ];
116
- return calls;
117
- };
118
-
119
- export const parseWstETHPriceCalls = (_ethPrice: string, wstETHrateAnswer: string, decimals: string) => {
120
- const ethPrice = new Dec(_ethPrice).div(1e8);
121
- const wstETHRate = getEthAmountForDecimals(wstETHrateAnswer, decimals);
122
- return { ethPrice, wstETHRate };
123
- };
124
-
125
- export const parseWeETHPriceCalls = (_ethPrice: string, weETHrateAnswer: string, decimals: string) => {
126
- const ethPrice = new Dec(_ethPrice).div(1e8);
127
- const weETHRate = getEthAmountForDecimals(weETHrateAnswer, decimals);
128
- return { ethPrice, weETHRate };
129
- };
130
-
131
- // this is a fixed version, the original version is above but requires to refactor comp v3 function, so it's easier to just copy the function for now
132
- export const getWstETHPriceFluid = async (client: PublicClient, network: NetworkNumber) => {
133
- const calls = getWstETHChainLinkPriceCalls(client, network);
134
- const results = await client.multicall({ contracts: calls });
135
- // @ts-ignore
136
- const { ethPrice, wstETHRate } = parseWstETHPriceCalls(results[0].result!.toString(), results[1].result[1]!.toString(), results[2].result!.toString());
137
-
138
- return new Dec(ethPrice).mul(wstETHRate).toString();
139
- };
140
-
141
- export const getBTCPriceForFluid = async (client: PublicClient, network: NetworkNumber) => {
142
- const contract = BTCPriceFeedContractViem(client, network);
143
- const price = await contract.read.latestAnswer();
144
- return new Dec(price).div(1e8).toString();
145
- };
146
-
147
- export const getEthPriceForFluid = async (client: PublicClient, network: NetworkNumber) => {
148
- const contract = ETHPriceFeedContractViem(client, network);
149
- const price = await contract.read.latestAnswer();
150
- return new Dec(price).div(1e8).toString();
151
- };
152
-
153
- // chainlink price feed available only on mainnet
154
- export const getChainlinkAssetAddress = (symbol: string, network: NetworkNumber): EthAddress => {
155
- // Chainlink only has BTC/USD feed so we use that for BTC derivatives
156
- if (['WBTC', 'RENBTC'].includes(symbol?.toUpperCase())) return '0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB';
157
- if (symbol?.toUpperCase() === 'WETH') return getAssetInfo('ETH').addresses[network] as EthAddress;
158
- return getAssetInfo(symbol).addresses[network] as EthAddress;
159
- };
1
+ import Dec from 'decimal.js';
2
+ import { getAssetInfo } from '@defisaver/tokens';
3
+ import { Client, PublicClient } from 'viem';
4
+ import {
5
+ COMPPriceFeedContractViem,
6
+ BTCPriceFeedContractViem,
7
+ ETHPriceFeedContractViem,
8
+ USDCPriceFeedContractViem,
9
+ WstETHPriceFeedContractViem,
10
+ WeETHPriceFeedContractViem,
11
+ } from '../contracts';
12
+ import { EthAddress, NetworkNumber } from '../types/common';
13
+ import { getEthAmountForDecimals } from './utils';
14
+
15
+ export const getEthPrice = async (client: Client) => {
16
+ const contract = ETHPriceFeedContractViem(client, NetworkNumber.Eth);
17
+ const price = await contract.read.latestAnswer();
18
+ return new Dec(price.toString()).div(1e8).toString();
19
+ };
20
+
21
+ export const getUSDCPrice = async (client: Client) => {
22
+ const contract = USDCPriceFeedContractViem(client, NetworkNumber.Eth);
23
+ const price = await contract.read.latestAnswer();
24
+ return new Dec(price.toString()).div(1e8).toString();
25
+ };
26
+
27
+ export const getCompPrice = async (client: Client) => {
28
+ const contract = COMPPriceFeedContractViem(client, NetworkNumber.Eth);
29
+ const price = await contract.read.latestAnswer();
30
+ return new Dec(price.toString()).div(1e8).toString();
31
+ };
32
+
33
+ export const getWstETHPrice = async (client: Client, network: NetworkNumber = NetworkNumber.Eth) => {
34
+ const wstETHFeedContract = WstETHPriceFeedContractViem(client, network);
35
+ const ethFeedContract = ETHPriceFeedContractViem(client, network);
36
+
37
+ const [ethPriceWei, wstETHRateWei, decimals] = await Promise.all([
38
+ ethFeedContract.read.latestAnswer(),
39
+ wstETHFeedContract.read.latestRoundData(),
40
+ wstETHFeedContract.read.decimals(),
41
+ ]);
42
+
43
+ const ethPrice = new Dec(ethPriceWei.toString()).div(1e8);
44
+
45
+ const wstETHRate = getEthAmountForDecimals(wstETHRateWei[1].toString(), decimals);
46
+
47
+ return new Dec(ethPrice).mul(wstETHRate).toString();
48
+ };
49
+
50
+ export const getWeETHPrice = async (client: Client, network: NetworkNumber = NetworkNumber.Eth) => {
51
+ const weETHFeedContract = WeETHPriceFeedContractViem(client, network);
52
+ const ethFeedContract = ETHPriceFeedContractViem(client, network);
53
+
54
+ const [ethPriceWei, weETHRateWei, decimals] = await Promise.all([
55
+ ethFeedContract.read.latestAnswer(),
56
+ weETHFeedContract.read.latestRoundData(),
57
+ weETHFeedContract.read.decimals(),
58
+ ]);
59
+
60
+ const ethPrice = new Dec(ethPriceWei.toString()).div(1e8);
61
+
62
+ const weETHRate = getEthAmountForDecimals(weETHRateWei[1].toString(), decimals);
63
+
64
+ return new Dec(ethPrice).mul(weETHRate).toString();
65
+ };
66
+
67
+ export const getWstETHChainLinkPriceCalls = (client: PublicClient, network: NetworkNumber) => {
68
+ const wstETHFeedContract = WstETHPriceFeedContractViem(client, network);
69
+ const ethFeedContract = ETHPriceFeedContractViem(client, network);
70
+ const calls = [
71
+ {
72
+ address: ethFeedContract.address,
73
+ abi: ethFeedContract.abi,
74
+ functionName: 'latestAnswer',
75
+ args: [],
76
+ },
77
+ {
78
+ address: wstETHFeedContract.address,
79
+ abi: wstETHFeedContract.abi,
80
+ functionName: 'latestRoundData',
81
+ args: [],
82
+ },
83
+ {
84
+ address: wstETHFeedContract.address,
85
+ abi: wstETHFeedContract.abi,
86
+ functionName: 'decimals',
87
+ args: [],
88
+ },
89
+ ];
90
+ return calls;
91
+ };
92
+
93
+ export const getWeETHChainLinkPriceCalls = (client: PublicClient, network: NetworkNumber) => {
94
+ const weETHFeedContract = WeETHPriceFeedContractViem(client, network);
95
+ const ethFeedContract = ETHPriceFeedContractViem(client, network);
96
+ const calls = [
97
+ {
98
+ address: ethFeedContract.address,
99
+ abi: ethFeedContract.abi,
100
+ functionName: 'latestAnswer',
101
+ args: [],
102
+ },
103
+ {
104
+ address: weETHFeedContract.address,
105
+ abi: weETHFeedContract.abi,
106
+ functionName: 'latestRoundData',
107
+ args: [],
108
+ },
109
+ {
110
+ address: weETHFeedContract.address,
111
+ abi: weETHFeedContract.abi,
112
+ functionName: 'decimals',
113
+ args: [],
114
+ },
115
+ ];
116
+ return calls;
117
+ };
118
+
119
+ export const parseWstETHPriceCalls = (_ethPrice: string, wstETHrateAnswer: string, decimals: string) => {
120
+ const ethPrice = new Dec(_ethPrice).div(1e8);
121
+ const wstETHRate = getEthAmountForDecimals(wstETHrateAnswer, decimals);
122
+ return { ethPrice, wstETHRate };
123
+ };
124
+
125
+ export const parseWeETHPriceCalls = (_ethPrice: string, weETHrateAnswer: string, decimals: string) => {
126
+ const ethPrice = new Dec(_ethPrice).div(1e8);
127
+ const weETHRate = getEthAmountForDecimals(weETHrateAnswer, decimals);
128
+ return { ethPrice, weETHRate };
129
+ };
130
+
131
+ // this is a fixed version, the original version is above but requires to refactor comp v3 function, so it's easier to just copy the function for now
132
+ export const getWstETHPriceFluid = async (client: PublicClient, network: NetworkNumber) => {
133
+ const calls = getWstETHChainLinkPriceCalls(client, network);
134
+ const results = await client.multicall({ contracts: calls });
135
+ // @ts-ignore
136
+ const { ethPrice, wstETHRate } = parseWstETHPriceCalls(results[0].result!.toString(), results[1].result[1]!.toString(), results[2].result!.toString());
137
+
138
+ return new Dec(ethPrice).mul(wstETHRate).toString();
139
+ };
140
+
141
+ export const getBTCPriceForFluid = async (client: PublicClient, network: NetworkNumber) => {
142
+ const contract = BTCPriceFeedContractViem(client, network);
143
+ const price = await contract.read.latestAnswer();
144
+ return new Dec(price).div(1e8).toString();
145
+ };
146
+
147
+ export const getEthPriceForFluid = async (client: PublicClient, network: NetworkNumber) => {
148
+ const contract = ETHPriceFeedContractViem(client, network);
149
+ const price = await contract.read.latestAnswer();
150
+ return new Dec(price).div(1e8).toString();
151
+ };
152
+
153
+ // chainlink price feed available only on mainnet
154
+ export const getChainlinkAssetAddress = (symbol: string, network: NetworkNumber): EthAddress => {
155
+ // Chainlink only has BTC/USD feed so we use that for BTC derivatives
156
+ if (['WBTC', 'RENBTC'].includes(symbol?.toUpperCase())) return '0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB';
157
+ if (symbol?.toUpperCase() === 'WETH') return getAssetInfo('ETH').addresses[network] as EthAddress;
158
+ return getAssetInfo(symbol).addresses[network] as EthAddress;
159
+ };
@@ -1,100 +1,100 @@
1
- import Dec from 'decimal.js';
2
- import { getAssetInfo, getAssetInfoByAddress } from '@defisaver/tokens';
3
- import { NetworkNumber } from '../types/common';
4
-
5
- export const isLayer2Network = (networkId: NetworkNumber) => [NetworkNumber.Opt, NetworkNumber.Arb, NetworkNumber.Base, NetworkNumber.Linea].includes(+networkId);
6
-
7
- export const addToObjectIf = (condition: any, item: any) => (condition ? item : {});
8
-
9
-
10
- export const addToArrayIf = (condition: any, ...items: any[]) => (condition ? items : []);
11
-
12
- export const ethToWeth = (maybeEth: any) => maybeEth?.replace(/^ETH$/, 'WETH');
13
-
14
- export const wethToEth = (maybeWeth: any) => maybeWeth?.replace(/^WETH$/, 'ETH');
15
-
16
- export const stEthToWstEth = (maybeStEth: any) => maybeStEth?.replace(/^stETH$/, 'wstETH');
17
-
18
- export const wstEthToStEth = (maybeStEth: any) => maybeStEth?.replace(/^wstETH$/, 'stETH');
19
-
20
- export const getAbiItem = (abi: any, methodName: string) => abi.find((i: any) => i.name === methodName);
21
-
22
- export const ADDRESS_REGEX = /0x[0-9a-fA-F]{40}/;
23
- export const isAddress = (address: string) => typeof address === 'string' && (new RegExp(ADDRESS_REGEX).test(address));
24
-
25
- export const compareAddresses = (addr1 = '', addr2 = '') => addr1.toLowerCase() === addr2.toLowerCase();
26
-
27
- export const getWeiAmountForDecimals = (amount: string | number, decimals: number) => new Dec(amount).mul(10 ** decimals).floor().toString();
28
-
29
- export const getEthAmountForDecimals = (amount: string | number, decimals: string | number) => new Dec(amount).div(10 ** +decimals).toString();
30
-
31
- export const handleWbtcLegacy = (asset: string) => (asset === 'WBTC Legacy' ? 'WBTC' : asset);
32
-
33
- export const wethToEthByAddress = (maybeWethAddr: string, chainId = NetworkNumber.Eth) => getAssetInfo(wethToEth(getAssetInfoByAddress(maybeWethAddr, chainId).symbol), chainId).address;
34
-
35
- export const ethToWethByAddress = (maybeEthAddr: string, chainId = NetworkNumber.Eth) => getAssetInfo(ethToWeth(getAssetInfoByAddress(maybeEthAddr, chainId).symbol), chainId).address;
36
-
37
- export const bytesToString = (hex: string) => Buffer.from(hex.replace(/^0x/, ''), 'hex')
38
- .toString()
39
- // eslint-disable-next-line no-control-regex
40
- .replace(/\x00/g, '');
41
-
42
- /**
43
- * Map an input value from one range (minInput, maxInput) to a value in another range (minOutput, maxOutput)
44
- */
45
- export const mapRange = (input: number | string, minInput: number | string, maxInput: number | string, minOutput:number | string, maxOutput: number | string) => {
46
- // slope = 1.0 * (output_end - output_start) / (input_end - input_start)
47
- const inputDiff = new Dec(maxInput).minus(minInput);
48
- const outputDiff = new Dec(maxOutput).minus(minOutput);
49
- const slope = new Dec(outputDiff).div(inputDiff);
50
-
51
- // output = output_start + slope * (input - input_start)
52
- return new Dec(minOutput).plus(new Dec(slope).mul(new Dec(input).minus(minInput))).toDP(2).toNumber();
53
- };
54
-
55
- // eslint-disable-next-line no-bitwise
56
- export const isEnabledOnBitmap = (bitmap: number, assetId: number) => (BigInt(bitmap) >> BigInt(assetId)) & BigInt(1);
57
-
58
- export const MAXUINT:string = '115792089237316195423570985008687907853269984665640564039457584007913129639935';
59
-
60
- export const isMaxuint = (amount: string) => compareAddresses(MAXUINT, amount);
61
-
62
- export const isMainnetNetwork = (network: NetworkNumber) => network === NetworkNumber.Eth;
63
-
64
- export const DEFAULT_TIMEOUT = 2000; // 2 seconds
65
-
66
- /**
67
- * Converts web3 hybrid response (that can be used as objects and arrays, so has duplicated values) to objects.
68
- * @param value
69
- */
70
- export const convertHybridArraysToObjects = (value: any): any => {
71
- if (Array.isArray(value)) {
72
- const keys = Object.keys(value);
73
- const hasNamedKeys = keys.some((key) => Number.isNaN(Number(key)));
74
-
75
- // If array has named keys, treat it as an object and keep only those
76
- if (hasNamedKeys) {
77
- const result: Record<string, any> = {};
78
- for (const [key, val] of Object.entries(value)) {
79
- if (Number.isNaN(Number(key))) {
80
- result[key] = convertHybridArraysToObjects(val);
81
- }
82
- }
83
- return result;
84
- }
85
-
86
- // Else, treat as a regular array
87
- return value.map(convertHybridArraysToObjects);
88
- }
89
-
90
- // If it's an object (but not an array), recurse into its values
91
- if (value && typeof value === 'object') {
92
- const result: Record<string, any> = {};
93
- for (const [key, val] of Object.entries(value)) {
94
- result[key] = convertHybridArraysToObjects(val);
95
- }
96
- return result;
97
- }
98
-
99
- return value;
1
+ import Dec from 'decimal.js';
2
+ import { getAssetInfo, getAssetInfoByAddress } from '@defisaver/tokens';
3
+ import { NetworkNumber } from '../types/common';
4
+
5
+ export const isLayer2Network = (networkId: NetworkNumber) => [NetworkNumber.Opt, NetworkNumber.Arb, NetworkNumber.Base, NetworkNumber.Linea].includes(+networkId);
6
+
7
+ export const addToObjectIf = (condition: any, item: any) => (condition ? item : {});
8
+
9
+
10
+ export const addToArrayIf = (condition: any, ...items: any[]) => (condition ? items : []);
11
+
12
+ export const ethToWeth = (maybeEth: any) => maybeEth?.replace(/^ETH$/, 'WETH');
13
+
14
+ export const wethToEth = (maybeWeth: any) => maybeWeth?.replace(/^WETH$/, 'ETH');
15
+
16
+ export const stEthToWstEth = (maybeStEth: any) => maybeStEth?.replace(/^stETH$/, 'wstETH');
17
+
18
+ export const wstEthToStEth = (maybeStEth: any) => maybeStEth?.replace(/^wstETH$/, 'stETH');
19
+
20
+ export const getAbiItem = (abi: any, methodName: string) => abi.find((i: any) => i.name === methodName);
21
+
22
+ export const ADDRESS_REGEX = /0x[0-9a-fA-F]{40}/;
23
+ export const isAddress = (address: string) => typeof address === 'string' && (new RegExp(ADDRESS_REGEX).test(address));
24
+
25
+ export const compareAddresses = (addr1 = '', addr2 = '') => addr1.toLowerCase() === addr2.toLowerCase();
26
+
27
+ export const getWeiAmountForDecimals = (amount: string | number, decimals: number) => new Dec(amount).mul(10 ** decimals).floor().toString();
28
+
29
+ export const getEthAmountForDecimals = (amount: string | number, decimals: string | number) => new Dec(amount).div(10 ** +decimals).toString();
30
+
31
+ export const handleWbtcLegacy = (asset: string) => (asset === 'WBTC Legacy' ? 'WBTC' : asset);
32
+
33
+ export const wethToEthByAddress = (maybeWethAddr: string, chainId = NetworkNumber.Eth) => getAssetInfo(wethToEth(getAssetInfoByAddress(maybeWethAddr, chainId).symbol), chainId).address;
34
+
35
+ export const ethToWethByAddress = (maybeEthAddr: string, chainId = NetworkNumber.Eth) => getAssetInfo(ethToWeth(getAssetInfoByAddress(maybeEthAddr, chainId).symbol), chainId).address;
36
+
37
+ export const bytesToString = (hex: string) => Buffer.from(hex.replace(/^0x/, ''), 'hex')
38
+ .toString()
39
+ // eslint-disable-next-line no-control-regex
40
+ .replace(/\x00/g, '');
41
+
42
+ /**
43
+ * Map an input value from one range (minInput, maxInput) to a value in another range (minOutput, maxOutput)
44
+ */
45
+ export const mapRange = (input: number | string, minInput: number | string, maxInput: number | string, minOutput:number | string, maxOutput: number | string) => {
46
+ // slope = 1.0 * (output_end - output_start) / (input_end - input_start)
47
+ const inputDiff = new Dec(maxInput).minus(minInput);
48
+ const outputDiff = new Dec(maxOutput).minus(minOutput);
49
+ const slope = new Dec(outputDiff).div(inputDiff);
50
+
51
+ // output = output_start + slope * (input - input_start)
52
+ return new Dec(minOutput).plus(new Dec(slope).mul(new Dec(input).minus(minInput))).toDP(2).toNumber();
53
+ };
54
+
55
+ // eslint-disable-next-line no-bitwise
56
+ export const isEnabledOnBitmap = (bitmap: number, assetId: number) => (BigInt(bitmap) >> BigInt(assetId)) & BigInt(1);
57
+
58
+ export const MAXUINT:string = '115792089237316195423570985008687907853269984665640564039457584007913129639935';
59
+
60
+ export const isMaxuint = (amount: string) => compareAddresses(MAXUINT, amount);
61
+
62
+ export const isMainnetNetwork = (network: NetworkNumber) => network === NetworkNumber.Eth;
63
+
64
+ export const DEFAULT_TIMEOUT = 2000; // 2 seconds
65
+
66
+ /**
67
+ * Converts web3 hybrid response (that can be used as objects and arrays, so has duplicated values) to objects.
68
+ * @param value
69
+ */
70
+ export const convertHybridArraysToObjects = (value: any): any => {
71
+ if (Array.isArray(value)) {
72
+ const keys = Object.keys(value);
73
+ const hasNamedKeys = keys.some((key) => Number.isNaN(Number(key)));
74
+
75
+ // If array has named keys, treat it as an object and keep only those
76
+ if (hasNamedKeys) {
77
+ const result: Record<string, any> = {};
78
+ for (const [key, val] of Object.entries(value)) {
79
+ if (Number.isNaN(Number(key))) {
80
+ result[key] = convertHybridArraysToObjects(val);
81
+ }
82
+ }
83
+ return result;
84
+ }
85
+
86
+ // Else, treat as a regular array
87
+ return value.map(convertHybridArraysToObjects);
88
+ }
89
+
90
+ // If it's an object (but not an array), recurse into its values
91
+ if (value && typeof value === 'object') {
92
+ const result: Record<string, any> = {};
93
+ for (const [key, val] of Object.entries(value)) {
94
+ result[key] = convertHybridArraysToObjects(val);
95
+ }
96
+ return result;
97
+ }
98
+
99
+ return value;
100
100
  };
@@ -1,33 +1,33 @@
1
- import { createPublicClient, custom } from 'viem';
2
- import {
3
- arbitrum, base, mainnet, optimism, linea,
4
- } from 'viem/chains';
5
- import { Blockish, EthereumProvider, NetworkNumber } from '../types/common';
6
-
7
- export const getViemChain = (network: NetworkNumber) => {
8
- switch (network) {
9
- case NetworkNumber.Eth:
10
- return mainnet;
11
- case NetworkNumber.Opt:
12
- return optimism;
13
- case NetworkNumber.Arb:
14
- return arbitrum;
15
- case NetworkNumber.Base:
16
- return base;
17
- case NetworkNumber.Linea:
18
- return linea;
19
- default:
20
- throw new Error(`Unsupported network: ${network}`);
21
- }
22
- };
23
-
24
- export const getViemProvider = (provider: EthereumProvider, network: NetworkNumber, options?: any) => createPublicClient({
25
- transport: custom(provider),
26
- chain: getViemChain(network),
27
- ...options,
28
- });
29
-
30
- export const setViemBlockNumber = (block: Blockish) => {
31
- if (block === 'latest') return {};
32
- return { blockNumber: BigInt(block) };
1
+ import { createPublicClient, custom } from 'viem';
2
+ import {
3
+ arbitrum, base, mainnet, optimism, linea,
4
+ } from 'viem/chains';
5
+ import { Blockish, EthereumProvider, NetworkNumber } from '../types/common';
6
+
7
+ export const getViemChain = (network: NetworkNumber) => {
8
+ switch (network) {
9
+ case NetworkNumber.Eth:
10
+ return mainnet;
11
+ case NetworkNumber.Opt:
12
+ return optimism;
13
+ case NetworkNumber.Arb:
14
+ return arbitrum;
15
+ case NetworkNumber.Base:
16
+ return base;
17
+ case NetworkNumber.Linea:
18
+ return linea;
19
+ default:
20
+ throw new Error(`Unsupported network: ${network}`);
21
+ }
22
+ };
23
+
24
+ export const getViemProvider = (provider: EthereumProvider, network: NetworkNumber, options?: any) => createPublicClient({
25
+ transport: custom(provider),
26
+ chain: getViemChain(network),
27
+ ...options,
28
+ });
29
+
30
+ export const setViemBlockNumber = (block: Blockish) => {
31
+ if (block === 'latest') return {};
32
+ return { blockNumber: BigInt(block) };
33
33
  };
package/src/setup.ts CHANGED
@@ -1,8 +1,8 @@
1
- import Decimal from 'decimal.js';
2
-
3
- Decimal.set({
4
- rounding: Decimal.ROUND_DOWN,
5
- toExpPos: 9e15,
6
- toExpNeg: -9e15,
7
- precision: 50,
8
- });
1
+ import Decimal from 'decimal.js';
2
+
3
+ Decimal.set({
4
+ rounding: Decimal.ROUND_DOWN,
5
+ toExpPos: 9e15,
6
+ toExpNeg: -9e15,
7
+ precision: 50,
8
+ });