erc20 0.0.16 → 0.0.18
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.
- checksums.yaml +4 -4
- data/lib/erc20/erc20.rb +1 -1
- data/lib/erc20/fake_wallet.rb +18 -0
- data/lib/erc20/wallet.rb +42 -17
- data/test/erc20/test_fake_wallet.rb +16 -0
- data/test/erc20/test_wallet.rb +40 -3
- metadata +1 -1
    
        checksums.yaml
    CHANGED
    
    | @@ -1,7 +1,7 @@ | |
| 1 1 | 
             
            ---
         | 
| 2 2 | 
             
            SHA256:
         | 
| 3 | 
            -
              metadata.gz:  | 
| 4 | 
            -
              data.tar.gz:  | 
| 3 | 
            +
              metadata.gz: 37aa39344c429cfd2153f004bd2a92730cd3a22c4f11c04e31573d36a98b2045
         | 
| 4 | 
            +
              data.tar.gz: c64d6baecf8256cce26bad81f16608b01098b1c581a1370dedb05b9761a8a6e2
         | 
| 5 5 | 
             
            SHA512:
         | 
| 6 | 
            -
              metadata.gz:  | 
| 7 | 
            -
              data.tar.gz:  | 
| 6 | 
            +
              metadata.gz: 030bd6e0734bef65820b1c0a63237eed014bc1e1291e7a98f2e8ffe17bffab6aaa3d6cc9685f0901002512376c6ac2bece5d9c564249259700df82ba2bb609c0
         | 
| 7 | 
            +
              data.tar.gz: 5836008fa63ea04bc68f7e8cc670f18b17ec887be606fedf1315e1d6e94d6be2b21859693c173d7ecd56e173fcd13f4598e08099fca941fb1bedd6367dc16e80
         | 
    
        data/lib/erc20/erc20.rb
    CHANGED
    
    
    
        data/lib/erc20/fake_wallet.rb
    CHANGED
    
    | @@ -59,6 +59,24 @@ class ERC20::FakeWallet | |
| 59 59 | 
             
                42_000_000
         | 
| 60 60 | 
             
              end
         | 
| 61 61 |  | 
| 62 | 
            +
              # How much ETH gas is required in order to send this ETH transaction.
         | 
| 63 | 
            +
              #
         | 
| 64 | 
            +
              # @param [String] _from The departing address, in hex
         | 
| 65 | 
            +
              # @param [String] _to Arriving address, in hex
         | 
| 66 | 
            +
              # @return [Integer] How many ETH required
         | 
| 67 | 
            +
              def eth_gas_required(_from, _to)
         | 
| 68 | 
            +
                55_000
         | 
| 69 | 
            +
              end
         | 
| 70 | 
            +
             | 
| 71 | 
            +
              # How much ETH gas is required in order to send this ERC20 transaction.
         | 
| 72 | 
            +
              #
         | 
| 73 | 
            +
              # @param [String] _from The departing address, in hex
         | 
| 74 | 
            +
              # @param [String] _to Arriving address, in hex
         | 
| 75 | 
            +
              # @return [Integer] How many ETH required
         | 
| 76 | 
            +
              def gas_required(_from, _to)
         | 
| 77 | 
            +
                66_000
         | 
| 78 | 
            +
              end
         | 
| 79 | 
            +
             | 
| 62 80 | 
             
              # Send a single ERC20 payment from a private address to a public one.
         | 
| 63 81 | 
             
              #
         | 
| 64 82 | 
             
              # @param [String] _priv Private key, in hex
         | 
    
        data/lib/erc20/wallet.rb
    CHANGED
    
    | @@ -159,6 +159,24 @@ class ERC20::Wallet | |
| 159 159 | 
             
                b
         | 
| 160 160 | 
             
              end
         | 
| 161 161 |  | 
| 162 | 
            +
              # How much ETH gas is required in order to send ERC20 transaction.
         | 
| 163 | 
            +
              #
         | 
| 164 | 
            +
              # @param [String] from The departing address, in hex
         | 
| 165 | 
            +
              # @param [String] to Arriving address, in hex (it's OK to skip it)
         | 
| 166 | 
            +
              # @return [Integer] How many ETH required
         | 
| 167 | 
            +
              def gas_required(from, to = from)
         | 
| 168 | 
            +
                gas_estimate(from, to, to_pay_data(from, 100_000))
         | 
| 169 | 
            +
              end
         | 
| 170 | 
            +
             | 
| 171 | 
            +
              # How much ETH gas is required in order to send this ETH transaction.
         | 
| 172 | 
            +
              #
         | 
| 173 | 
            +
              # @param [String] from The departing address, in hex
         | 
| 174 | 
            +
              # @param [String] to Arriving address, in hex (it's OK to skip it)
         | 
| 175 | 
            +
              # @return [Integer] How many ETH required
         | 
| 176 | 
            +
              def eth_gas_required(from, to = from)
         | 
| 177 | 
            +
                gas_estimate(from, to)
         | 
| 178 | 
            +
              end
         | 
| 179 | 
            +
             | 
| 162 180 | 
             
              # Send a single ERC20 payment from a private address to a public one.
         | 
| 163 181 | 
             
              #
         | 
| 164 182 | 
             
              # @param [String] priv Private key, in hex
         | 
| @@ -167,7 +185,7 @@ class ERC20::Wallet | |
| 167 185 | 
             
              # @param [Integer] gas_limit How much gas you're ready to spend
         | 
| 168 186 | 
             
              # @param [Integer] gas_price How much gas you pay per computation unit
         | 
| 169 187 | 
             
              # @return [String] Transaction hash
         | 
| 170 | 
            -
              def pay(priv, address, amount, gas_limit: nil, gas_price:  | 
| 188 | 
            +
              def pay(priv, address, amount, gas_limit: nil, gas_price: gas_best_price)
         | 
| 171 189 | 
             
                raise 'Private key can\'t be nil' unless priv
         | 
| 172 190 | 
             
                raise 'Private key must be a String' unless priv.is_a?(String)
         | 
| 173 191 | 
             
                raise 'Invalid format of private key' unless /^[0-9a-fA-F]{64}$/.match?(priv)
         | 
| @@ -185,25 +203,20 @@ class ERC20::Wallet | |
| 185 203 | 
             
                  raise 'Gas price must be an Integer' unless gas_price.is_a?(Integer)
         | 
| 186 204 | 
             
                  raise 'Gas price must be a positive Integer' unless gas_price.positive?
         | 
| 187 205 | 
             
                end
         | 
| 188 | 
            -
                func = 'a9059cbb' # transfer(address,uint256)
         | 
| 189 | 
            -
                to_clean = address.downcase.sub(/^0x/, '')
         | 
| 190 | 
            -
                to_padded = ('0' * (64 - to_clean.size)) + to_clean
         | 
| 191 | 
            -
                amt_hex = amount.to_s(16)
         | 
| 192 | 
            -
                amt_padded = ('0' * (64 - amt_hex.size)) + amt_hex
         | 
| 193 | 
            -
                data = "0x#{func}#{to_padded}#{amt_padded}"
         | 
| 194 206 | 
             
                key = Eth::Key.new(priv: priv)
         | 
| 195 207 | 
             
                from = key.address.to_s
         | 
| 208 | 
            +
                data = to_pay_data(address, amount)
         | 
| 196 209 | 
             
                tnx =
         | 
| 197 210 | 
             
                  @mutex.synchronize do
         | 
| 198 211 | 
             
                    nonce = jsonrpc.eth_getTransactionCount(from, 'pending').to_i(16)
         | 
| 199 212 | 
             
                    tx = Eth::Tx.new(
         | 
| 200 213 | 
             
                      {
         | 
| 201 214 | 
             
                        nonce:,
         | 
| 202 | 
            -
                        gas_price: gas_price | 
| 215 | 
            +
                        gas_price: gas_price,
         | 
| 203 216 | 
             
                        gas_limit: gas_limit || gas_estimate(from, @contract, data),
         | 
| 204 217 | 
             
                        to: @contract,
         | 
| 205 218 | 
             
                        value: 0,
         | 
| 206 | 
            -
                        data | 
| 219 | 
            +
                        data:,
         | 
| 207 220 | 
             
                        chain_id: @chain
         | 
| 208 221 | 
             
                      }
         | 
| 209 222 | 
             
                    )
         | 
| @@ -223,7 +236,7 @@ class ERC20::Wallet | |
| 223 236 | 
             
              # @param [Integer] gas_limit How much gas you're ready to spend
         | 
| 224 237 | 
             
              # @param [Integer] gas_price How much gas you pay per computation unit
         | 
| 225 238 | 
             
              # @return [String] Transaction hash
         | 
| 226 | 
            -
              def eth_pay(priv, address, amount, gas_limit: nil, gas_price:  | 
| 239 | 
            +
              def eth_pay(priv, address, amount, gas_limit: nil, gas_price: gas_best_price)
         | 
| 227 240 | 
             
                raise 'Private key can\'t be nil' unless priv
         | 
| 228 241 | 
             
                raise 'Private key must be a String' unless priv.is_a?(String)
         | 
| 229 242 | 
             
                raise 'Invalid format of private key' unless /^[0-9a-fA-F]{64}$/.match?(priv)
         | 
| @@ -243,7 +256,6 @@ class ERC20::Wallet | |
| 243 256 | 
             
                end
         | 
| 244 257 | 
             
                key = Eth::Key.new(priv: priv)
         | 
| 245 258 | 
             
                from = key.address.to_s
         | 
| 246 | 
            -
                data = ''
         | 
| 247 259 | 
             
                tnx =
         | 
| 248 260 | 
             
                  @mutex.synchronize do
         | 
| 249 261 | 
             
                    nonce = jsonrpc.eth_getTransactionCount(from, 'pending').to_i(16)
         | 
| @@ -251,11 +263,10 @@ class ERC20::Wallet | |
| 251 263 | 
             
                      {
         | 
| 252 264 | 
             
                        chain_id: @chain,
         | 
| 253 265 | 
             
                        nonce:,
         | 
| 254 | 
            -
                        gas_price: gas_price | 
| 255 | 
            -
                        gas_limit: gas_limit || gas_estimate(from, address | 
| 266 | 
            +
                        gas_price: gas_price,
         | 
| 267 | 
            +
                        gas_limit: gas_limit || gas_estimate(from, address),
         | 
| 256 268 | 
             
                        to: address,
         | 
| 257 | 
            -
                        value: amount | 
| 258 | 
            -
                        data:
         | 
| 269 | 
            +
                        value: amount
         | 
| 259 270 | 
             
                      }
         | 
| 260 271 | 
             
                    )
         | 
| 261 272 | 
             
                    tx.sign(key)
         | 
| @@ -415,8 +426,22 @@ class ERC20::Wallet | |
| 415 426 | 
             
                JSONRPC::Client.new(url, connection:)
         | 
| 416 427 | 
             
              end
         | 
| 417 428 |  | 
| 418 | 
            -
               | 
| 419 | 
            -
             | 
| 429 | 
            +
              # How much gas should be spent in order to send a transaction from one
         | 
| 430 | 
            +
              # public address to another public address, possible carrying some data
         | 
| 431 | 
            +
              # inside the transaction.
         | 
| 432 | 
            +
              def gas_estimate(from, to, data = '')
         | 
| 433 | 
            +
                gas = jsonrpc.eth_estimateGas({ from:, to:, data: }, 'latest').to_i(16)
         | 
| 434 | 
            +
                @log.debug("Estimated gas is #{gas} ETH#{data.empty? ? '' : ', for ERC20 transfer'}")
         | 
| 435 | 
            +
                gas
         | 
| 436 | 
            +
              end
         | 
| 437 | 
            +
             | 
| 438 | 
            +
              def to_pay_data(address, amount)
         | 
| 439 | 
            +
                func = 'a9059cbb' # transfer(address,uint256)
         | 
| 440 | 
            +
                to_clean = address.downcase.sub(/^0x/, '')
         | 
| 441 | 
            +
                to_padded = ('0' * (64 - to_clean.size)) + to_clean
         | 
| 442 | 
            +
                amt_hex = amount.to_s(16)
         | 
| 443 | 
            +
                amt_padded = ('0' * (64 - amt_hex.size)) + amt_hex
         | 
| 444 | 
            +
                "0x#{func}#{to_padded}#{amt_padded}"
         | 
| 420 445 | 
             
              end
         | 
| 421 446 |  | 
| 422 447 | 
             
              def gas_best_price
         | 
| @@ -38,6 +38,22 @@ require_relative '../test__helper' | |
| 38 38 | 
             
            # Copyright:: Copyright (c) 2025 Yegor Bugayenko
         | 
| 39 39 | 
             
            # License:: MIT
         | 
| 40 40 | 
             
            class TestFakeWallet < Minitest::Test
         | 
| 41 | 
            +
              def test_checks_gas_required
         | 
| 42 | 
            +
                b = ERC20::FakeWallet.new.gas_required(
         | 
| 43 | 
            +
                  '0xEB2fE8872A6f1eDb70a2632Effffffffffffffff',
         | 
| 44 | 
            +
                  '0xEB2fE8872A6f1eDb70a2632Effffffffffffffff'
         | 
| 45 | 
            +
                )
         | 
| 46 | 
            +
                refute_nil(b)
         | 
| 47 | 
            +
              end
         | 
| 48 | 
            +
             | 
| 49 | 
            +
              def test_checks_eth_gas_required
         | 
| 50 | 
            +
                b = ERC20::FakeWallet.new.eth_gas_required(
         | 
| 51 | 
            +
                  '0xEB2fE8872A6f1eDb70a2632Effffffffffffffff',
         | 
| 52 | 
            +
                  '0xEB2fE8872A6f1eDb70a2632Effffffffffffffff'
         | 
| 53 | 
            +
                )
         | 
| 54 | 
            +
                refute_nil(b)
         | 
| 55 | 
            +
              end
         | 
| 56 | 
            +
             | 
| 41 57 | 
             
              def test_checks_fake_balance
         | 
| 42 58 | 
             
                b = ERC20::FakeWallet.new.balance('0xEB2fE8872A6f1eDb70a2632Effffffffffffffff')
         | 
| 43 59 | 
             
                refute_nil(b)
         | 
    
        data/test/erc20/test_wallet.rb
    CHANGED
    
    | @@ -38,7 +38,7 @@ require_relative '../test__helper' | |
| 38 38 | 
             
            # Copyright:: Copyright (c) 2025 Yegor Bugayenko
         | 
| 39 39 | 
             
            # License:: MIT
         | 
| 40 40 | 
             
            class TestWallet < Minitest::Test
         | 
| 41 | 
            -
              # At this address, in Etherium mainnet, there are $8 USDT. I won't
         | 
| 41 | 
            +
              # At this address, in Etherium mainnet, there are $8 USDT and 0.0042 ETH. I won't
         | 
| 42 42 | 
             
              # move them anyway, that's why tests can use this address forever.
         | 
| 43 43 | 
             
              STABLE = '0x7232148927F8a580053792f44D4d59d40Fd00ABD'
         | 
| 44 44 |  | 
| @@ -51,13 +51,13 @@ class TestWallet < Minitest::Test | |
| 51 51 | 
             
              def test_checks_balance_on_mainnet
         | 
| 52 52 | 
             
                b = mainnet.balance(STABLE)
         | 
| 53 53 | 
             
                refute_nil(b)
         | 
| 54 | 
            -
                assert_equal(8_000_000, b)
         | 
| 54 | 
            +
                assert_equal(8_000_000, b) # this is $8 USDT
         | 
| 55 55 | 
             
              end
         | 
| 56 56 |  | 
| 57 57 | 
             
              def test_checks_eth_balance_on_mainnet
         | 
| 58 58 | 
             
                b = mainnet.eth_balance(STABLE)
         | 
| 59 59 | 
             
                refute_nil(b)
         | 
| 60 | 
            -
                assert_equal( | 
| 60 | 
            +
                assert_equal(4_200_000_000_000_000, b) # this is 0.0042 ETH
         | 
| 61 61 | 
             
              end
         | 
| 62 62 |  | 
| 63 63 | 
             
              def test_checks_balance_of_absent_address
         | 
| @@ -67,6 +67,18 @@ class TestWallet < Minitest::Test | |
| 67 67 | 
             
                assert_equal(0, b)
         | 
| 68 68 | 
             
              end
         | 
| 69 69 |  | 
| 70 | 
            +
              def test_checks_gas_required_on_mainnet
         | 
| 71 | 
            +
                b = mainnet.gas_required(STABLE, Eth::Key.new(priv: JEFF).address.to_s)
         | 
| 72 | 
            +
                refute_nil(b)
         | 
| 73 | 
            +
                assert_predicate(b, :positive?)
         | 
| 74 | 
            +
              end
         | 
| 75 | 
            +
             | 
| 76 | 
            +
              def test_checks_same_address_gas_required_on_mainnet
         | 
| 77 | 
            +
                b = mainnet.gas_required(STABLE, STABLE)
         | 
| 78 | 
            +
                refute_nil(b)
         | 
| 79 | 
            +
                assert_predicate(b, :positive?)
         | 
| 80 | 
            +
              end
         | 
| 81 | 
            +
             | 
| 70 82 | 
             
              def test_fails_with_invalid_infura_key
         | 
| 71 83 | 
             
                skip('Apparently, even with invalid key, Infura returns balance')
         | 
| 72 84 | 
             
                w = ERC20::Wallet.new(
         | 
| @@ -94,6 +106,31 @@ class TestWallet < Minitest::Test | |
| 94 106 | 
             
                assert_predicate(b, :zero?)
         | 
| 95 107 | 
             
              end
         | 
| 96 108 |  | 
| 109 | 
            +
              def test_checks_gas_required_on_hardhat
         | 
| 110 | 
            +
                on_hardhat do |wallet|
         | 
| 111 | 
            +
                  b1 = wallet.gas_required(
         | 
| 112 | 
            +
                    Eth::Key.new(priv: JEFF).address.to_s,
         | 
| 113 | 
            +
                    Eth::Key.new(priv: WALTER).address.to_s
         | 
| 114 | 
            +
                  )
         | 
| 115 | 
            +
                  assert_equal(21_597, b1)
         | 
| 116 | 
            +
                  b2 = wallet.gas_required(
         | 
| 117 | 
            +
                    Eth::Key.new(priv: JEFF).address.to_s,
         | 
| 118 | 
            +
                    Eth::Key.new(priv: JEFF).address.to_s
         | 
| 119 | 
            +
                  )
         | 
| 120 | 
            +
                  assert_equal(b1, b2)
         | 
| 121 | 
            +
                end
         | 
| 122 | 
            +
              end
         | 
| 123 | 
            +
             | 
| 124 | 
            +
              def test_checks_eth_gas_required_on_hardhat
         | 
| 125 | 
            +
                on_hardhat do |wallet|
         | 
| 126 | 
            +
                  b = wallet.eth_gas_required(
         | 
| 127 | 
            +
                    Eth::Key.new(priv: JEFF).address.to_s,
         | 
| 128 | 
            +
                    Eth::Key.new(priv: WALTER).address.to_s
         | 
| 129 | 
            +
                  )
         | 
| 130 | 
            +
                  assert_equal(21_001, b)
         | 
| 131 | 
            +
                end
         | 
| 132 | 
            +
              end
         | 
| 133 | 
            +
             | 
| 97 134 | 
             
              def test_checks_balance_on_hardhat
         | 
| 98 135 | 
             
                on_hardhat do |wallet|
         | 
| 99 136 | 
             
                  b = wallet.balance(Eth::Key.new(priv: JEFF).address.to_s)
         |