@exodus/ethereum-api 2.13.0 → 2.13.2

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.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@exodus/ethereum-api",
3
- "version": "2.13.0",
3
+ "version": "2.13.2",
4
4
  "description": "Ethereum Api",
5
5
  "main": "src/index.js",
6
6
  "author": "Exodus Movement, Inc.",
@@ -12,7 +12,7 @@
12
12
  "dependencies": {
13
13
  "@exodus/asset-lib": "^3.5.4",
14
14
  "@exodus/crypto": "^1.0.0-rc.0",
15
- "@exodus/ethereum-lib": "^2.13.4",
15
+ "@exodus/ethereum-lib": "^2.13.6",
16
16
  "@exodus/ethereumjs-util": "^7.1.0-exodus.6",
17
17
  "@exodus/simple-retry": "^0.0.6",
18
18
  "fetchival": "0.3.3",
@@ -28,5 +28,5 @@
28
28
  "@exodus/assets-base": "^8.0.136",
29
29
  "@exodus/models": "^8.7.2"
30
30
  },
31
- "gitHead": "4fddf195ead826e4f407841a06e00c794cc41952"
31
+ "gitHead": "f1d39b200c106794d81bd19b7e41073eeec15cba"
32
32
  }
@@ -0,0 +1,38 @@
1
+ import { isRpcBalanceAsset } from '@exodus/ethereum-lib'
2
+
3
+ const fixBalance = ({ txLog, balance }) => {
4
+ for (const tx of txLog) {
5
+ // TODO: pending can only be less than a few minutes old, we can only search the latest txs to improve performance
6
+ if (tx.sent && tx.pending && !tx.error) {
7
+ // coinAmount is negative for sent tx
8
+ balance = balance.sub(tx.coinAmount.abs())
9
+ if (tx.coinAmount.unitType.equals(tx.feeAmount.unitType)) {
10
+ balance = balance.sub(tx.feeAmount)
11
+ }
12
+ }
13
+ }
14
+ return balance
15
+ }
16
+
17
+ /**
18
+ * Api method to return the balance based on either account state balances or tx history.
19
+ *
20
+ * @param asset the asset to get the balances
21
+ * @param txLog the txLog when the balance is transaction based
22
+ * @param accountState the account state when the balance is loaded from RPC
23
+ * @returns {{balance}|null} an object with the balance or null if the balance is unknown/zero
24
+ */
25
+ export const getBalances = ({ asset, txLog, accountState }) => {
26
+ if (isRpcBalanceAsset(asset)) {
27
+ const balance =
28
+ asset.baseAsset.name === asset.name
29
+ ? accountState?.balance
30
+ : accountState?.tokenBalances?.[asset.name]
31
+ return balance && !balance.isZero ? { balance: fixBalance({ txLog, balance }) } : null
32
+ }
33
+ return txLog.size
34
+ ? {
35
+ balance: txLog.getMutations().slice(-1)[0].balance,
36
+ }
37
+ : null
38
+ }
package/src/index.js CHANGED
@@ -6,3 +6,5 @@ export * from './fee-monitor'
6
6
  export * from './gas-estimation'
7
7
  export * from './exodus-eth-server'
8
8
  export * from './tx-log'
9
+ export * from './get-balances'
10
+ export * from './staking'
@@ -0,0 +1 @@
1
+ export { MaticStaking } from './matic-staking'
@@ -0,0 +1,159 @@
1
+ import { createContract } from '@exodus/ethereum-lib'
2
+ import { eth } from '../exodus-eth-server'
3
+ import { retry } from '@exodus/simple-retry'
4
+ import { bufferToHex } from '@exodus/ethereumjs-util'
5
+
6
+ export const EVERSTAKE_VALIDATOR_ADDR = '0xf30cf4ed712d3734161fdaab5b1dbb49fd2d0e5c'
7
+ export const STAKING_MANAGER_ADDR = '0x5e3ef299fddf15eaa0432e6e66473ace8c13d908'
8
+
9
+ const RETRY_DELAYS = ['10s']
10
+
11
+ export class MaticStaking {
12
+ constructor(validatorId = EVERSTAKE_VALIDATOR_ADDR, stakeManagerAddr = STAKING_MANAGER_ADDR) {
13
+ this.validatorShareContract = createContract(validatorId, 'maticValidatorShare')
14
+ this.stakingManagerContract = createContract(stakeManagerAddr, 'maticStakingManager')
15
+ }
16
+
17
+ buildTxData = (contract, method, ...args) => {
18
+ const txData = contract[method].build(...args)
19
+ return txData
20
+ }
21
+
22
+ processRead = (contract, method, ...args) => {
23
+ const callData = this.buildTxData(contract, method, ...args)
24
+ const data = {
25
+ data: bufferToHex(callData),
26
+ to: contract.address,
27
+ tag: 'latest',
28
+ }
29
+
30
+ return retry(eth.ethCall, { delayTimesMs: RETRY_DELAYS })(data)
31
+ }
32
+
33
+ /**
34
+ * Get current checkpoint
35
+ * @returns {number}
36
+ */
37
+ getCurrentCheckpoint = async () => {
38
+ const currentEpoch = await this.processRead(this.stakingManagerContract, 'epoch')
39
+ return parseInt(currentEpoch, 16)
40
+ }
41
+
42
+ getCurrentUnbondNonce = async (address) => {
43
+ const result = await this.processRead(this.validatorShareContract, 'unbondNonces', address)
44
+ return parseInt(result, 16)
45
+ }
46
+
47
+ /**
48
+ * Get unbond info by supplying unbondNonces
49
+ * @param {string} address
50
+ * @param {number} nonce
51
+ * @returns {number}
52
+ */
53
+ getUnboundInfo = async (address, nonce) => {
54
+ let _nonce = nonce
55
+ if (!Number.isInteger(_nonce)) {
56
+ _nonce = await this.getCurrentUnbondNonce(address)
57
+ }
58
+
59
+ const unboundInfo = await this.processRead(
60
+ this.validatorShareContract,
61
+ 'unbonds_new',
62
+ address,
63
+ _nonce
64
+ )
65
+ const [withdrawEpoch, shares] = this._removeHexPrefix(unboundInfo).match(/.{1,64}/g) || []
66
+ return {
67
+ withdrawEpoch: parseInt(withdrawEpoch, 16),
68
+ shares: parseInt(shares, 16),
69
+ }
70
+ }
71
+
72
+ /**
73
+ * Get current Liquid rewards
74
+ * @param {string} address
75
+ * @returns {number}
76
+ */
77
+ getLiquidRewards = async (address) => {
78
+ const currentEpoch = await this.processRead(
79
+ this.validatorShareContract,
80
+ 'getLiquidRewards',
81
+ address
82
+ )
83
+ return parseInt(currentEpoch, 16)
84
+ }
85
+
86
+ /**
87
+ * Get total staked amount
88
+ * @param {string} address
89
+ * @returns {number}
90
+ */
91
+ getTotalStake = async (address) => {
92
+ const stakeInfo = await this.processRead(this.validatorShareContract, 'getTotalStake', address)
93
+ const [amount] = this._removeHexPrefix(stakeInfo).match(/.{1,64}/g) || []
94
+ return parseInt(amount, 16)
95
+ }
96
+
97
+ /**
98
+ * Get tx data of method to restake the earned rewards
99
+ */
100
+ restakeReward = () => {
101
+ return this.buildTxData(this.validatorShareContract, 'restake')
102
+ }
103
+
104
+ /**
105
+ * Get tx data of method to withdraw earned rewards
106
+ */
107
+ withdrawRewards = () => {
108
+ return this.buildTxData(this.validatorShareContract, 'withdrawRewards')
109
+ }
110
+
111
+ /**
112
+ * Get tx data of method to delegate (stake) amount to validator
113
+ * internally it calls method - **buyVoucher**
114
+ * @param {string} param.amount
115
+ */
116
+ delegateAmount = ({ amount }) => {
117
+ return this.buildTxData(this.validatorShareContract, 'buyVoucher', amount, 0)
118
+ }
119
+
120
+ /**
121
+ * Get tx data of method to unstake delegated amount
122
+ * internally it calls - **sellVoucher_new**
123
+ * @param {string} param.amount
124
+ * @param {string} param.maximumSharesToBurn
125
+ * @returns {string}
126
+ */
127
+ unbond = ({ amount, maximumSharesToBurn }) => {
128
+ const _maximumSharesToBurn = maximumSharesToBurn || amount
129
+ return this.buildTxData(
130
+ this.validatorShareContract,
131
+ 'sellVoucher_new',
132
+ amount,
133
+ _maximumSharesToBurn
134
+ )
135
+ }
136
+
137
+ /**
138
+ * Get tx data of method to claim the delegated amount by supplying nonce
139
+ * internally it calls method - **unstakeClaimTokens_new**
140
+ * @param {number} nonce
141
+ * @returns {string}
142
+ */
143
+ claimStake = (nonce) => {
144
+ return this.buildTxData(this.validatorShareContract, 'unstakeClaimTokens_new', nonce)
145
+ }
146
+
147
+ /**
148
+ * Remove 0x prefix of a hex string
149
+ * @param {string} str
150
+ * @returns {string}
151
+ */
152
+ _removeHexPrefix = (str) => {
153
+ if (typeof str !== 'string' || str === '') {
154
+ return str
155
+ }
156
+
157
+ return str.startsWith('0x') ? str.slice(2) : str
158
+ }
159
+ }
@@ -224,9 +224,8 @@ export class EthereumMonitor extends BaseMonitor {
224
224
  .filter((token) => isRpcBalanceAsset(token) && token.contract.address)
225
225
  .map(async (token) => {
226
226
  const { confirmed } = await server.balanceOf(ourWalletAddress, token.contract.address)
227
- const value = confirmed[token.contract.address]
228
- const balance = token.currency.baseUnit(value || 0)
229
- return { [token.name]: balance }
227
+ const value = token.currency.baseUnit(confirmed[token.contract.address] || 0)
228
+ return value.isZero ? {} : { [token.name]: value }
230
229
  })
231
230
  ))
232
231
  )