erc20 0.0.12 → 0.0.14

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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 0c9c69024f80c9041c0f80364a9e8129cf2d7802f9320576fbf007b9b3f32e9f
4
- data.tar.gz: 00f5eba51cde139b5a4c697253fc285152c52c06c8927cbdac34deb40863702b
3
+ metadata.gz: ac1bd8d3ffe6bf19ceb44174514dca9abb57746110c864821de775b2f11f955a
4
+ data.tar.gz: 7cc7b1d4fe20626629ea99ca5bb6cbb02686c872e36106c4c05fab846ad9b24b
5
5
  SHA512:
6
- metadata.gz: 10d041c08ff2cbcb1e1fe7521980d26919b6b4926e1506a6341bd36280886cf26ee9d9192c4b2b3c4bf6e6e9237251a9fc3d4bda3430cdaaf54f229146b7febf
7
- data.tar.gz: 802187ea0f8acc530b22f9d5518b60e534a8a633eac114c0f5c682a8cc1b079edae4d130fc7ac1b6d52c3f24b443c2f3939b4727d61524ddca6fe2d94b90d726
6
+ metadata.gz: 7212c53bce48a094d0eaa8359bed81cad82e81f883caa47da2da4851cff7c6d3956036a3919265925f540b125cede427f5a9c506a60e3b002469bed22096695a
7
+ data.tar.gz: 7a80add54d6ecf810bff74a265cf4dca8ae8b99ae8c8f9e046cfec537bc9256d7a6127159d52d0b04001a7c3b17f9895c4b64814e7d9b058789e2c267b648a21
data/.rubocop.yml CHANGED
@@ -27,10 +27,10 @@ AllCops:
27
27
  TargetRubyVersion: 3.2
28
28
  SuggestExtensions: false
29
29
  NewCops: enable
30
- require:
31
- - rubocop-minitest
30
+ plugins:
32
31
  - rubocop-performance
33
32
  - rubocop-rake
33
+ - rubocop-minitest
34
34
  Minitest/EmptyLineBeforeAssertionMethods:
35
35
  Enabled: false
36
36
  Gemspec/RequiredRubyVersion:
data/Gemfile CHANGED
@@ -34,9 +34,9 @@ gem 'qbash', '>0', require: false
34
34
  gem 'rake', '13.2.1', require: false
35
35
  gem 'random-port', '>0', require: false
36
36
  gem 'rspec-rails', '7.1.1', require: false
37
- gem 'rubocop', '1.71.2', require: false
38
- gem 'rubocop-minitest', '0.36.0', require: false
39
- gem 'rubocop-performance', '1.23.1', require: false
37
+ gem 'rubocop', '1.72.1', require: false
38
+ gem 'rubocop-minitest', '0.37.1', require: false
39
+ gem 'rubocop-performance', '1.24.0', require: false
40
40
  gem 'rubocop-rake', '>0', require: false
41
41
  gem 'rubocop-rspec', '3.4.0', require: false
42
42
  gem 'simplecov', '0.22.0', require: false
data/Gemfile.lock CHANGED
@@ -103,6 +103,7 @@ GEM
103
103
  keccak (1.3.2)
104
104
  konstructor (1.0.2)
105
105
  language_server-protocol (3.17.0.4)
106
+ lint_roller (1.1.0)
106
107
  logger (1.6.6)
107
108
  loofah (2.24.0)
108
109
  crass (~> 1.0.2)
@@ -201,9 +202,10 @@ GEM
201
202
  rspec-mocks (~> 3.13)
202
203
  rspec-support (~> 3.13)
203
204
  rspec-support (3.13.2)
204
- rubocop (1.71.2)
205
+ rubocop (1.72.1)
205
206
  json (~> 2.3)
206
- language_server-protocol (>= 3.17.0)
207
+ language_server-protocol (~> 3.17.0.2)
208
+ lint_roller (~> 1.1.0)
207
209
  parallel (~> 1.10)
208
210
  parser (>= 3.3.0.2)
209
211
  rainbow (>= 2.2.2, < 4.0)
@@ -213,14 +215,17 @@ GEM
213
215
  unicode-display_width (>= 2.4.0, < 4.0)
214
216
  rubocop-ast (1.38.0)
215
217
  parser (>= 3.3.1.0)
216
- rubocop-minitest (0.36.0)
217
- rubocop (>= 1.61, < 2.0)
218
- rubocop-ast (>= 1.31.1, < 2.0)
219
- rubocop-performance (1.23.1)
220
- rubocop (>= 1.48.1, < 2.0)
221
- rubocop-ast (>= 1.31.1, < 2.0)
222
- rubocop-rake (0.6.0)
223
- rubocop (~> 1.0)
218
+ rubocop-minitest (0.37.1)
219
+ lint_roller (~> 1.1)
220
+ rubocop (>= 1.72.1, < 2.0)
221
+ rubocop-ast (>= 1.38.0, < 2.0)
222
+ rubocop-performance (1.24.0)
223
+ lint_roller (~> 1.1)
224
+ rubocop (>= 1.72.1, < 2.0)
225
+ rubocop-ast (>= 1.38.0, < 2.0)
226
+ rubocop-rake (0.7.0)
227
+ lint_roller (~> 1.1)
228
+ rubocop (>= 1.72.1)
224
229
  rubocop-rspec (3.4.0)
225
230
  rubocop (~> 1.61)
226
231
  ruby-progressbar (1.13.0)
@@ -238,7 +243,7 @@ GEM
238
243
  simplecov (~> 0.19)
239
244
  simplecov-html (0.13.1)
240
245
  simplecov_json_formatter (0.1.4)
241
- stringio (3.1.2)
246
+ stringio (3.1.3)
242
247
  tago (0.0.2)
243
248
  thor (1.3.2)
244
249
  threads (0.4.1)
@@ -282,9 +287,9 @@ DEPENDENCIES
282
287
  rake (= 13.2.1)
283
288
  random-port (> 0)
284
289
  rspec-rails (= 7.1.1)
285
- rubocop (= 1.71.2)
286
- rubocop-minitest (= 0.36.0)
287
- rubocop-performance (= 1.23.1)
290
+ rubocop (= 1.72.1)
291
+ rubocop-minitest (= 0.37.1)
292
+ rubocop-performance (= 1.24.0)
288
293
  rubocop-rake (> 0)
289
294
  rubocop-rspec (= 3.4.0)
290
295
  simplecov (= 0.22.0)
data/lib/erc20/erc20.rb CHANGED
@@ -42,5 +42,5 @@
42
42
  # License:: MIT
43
43
  module ERC20
44
44
  # Current version of the gem (changed by the +.rultor.yml+ on every release)
45
- VERSION = '0.0.12'
45
+ VERSION = '0.0.14'
46
46
  end
data/lib/erc20/wallet.rb CHANGED
@@ -93,13 +93,30 @@ class ERC20::Wallet
93
93
  def initialize(contract: USDT, chain: 1, log: $stdout,
94
94
  host: nil, port: 443, http_path: '/', ws_path: '/',
95
95
  ssl: true, proxy: nil)
96
+ raise 'Contract can\'t be nil' unless contract
97
+ raise 'Contract must be a String' unless contract.is_a?(String)
98
+ raise 'Invalid format of the contract' unless /^0x[0-9a-fA-F]{40}$/.match?(contract)
96
99
  @contract = contract
100
+ raise 'Host can\'t be nil' unless host
101
+ raise 'Host must be a String' unless host.is_a?(String)
97
102
  @host = host
103
+ raise 'Port can\'t be nil' unless port
104
+ raise 'Port must be an Integer' unless port.is_a?(Integer)
105
+ raise 'Port must be a positive Integer' unless port.positive?
98
106
  @port = port
107
+ raise 'Ssl can\'t be nil' if ssl.nil?
99
108
  @ssl = ssl
109
+ raise 'Http_path can\'t be nil' unless http_path
110
+ raise 'Http_path must be a String' unless http_path.is_a?(String)
100
111
  @http_path = http_path
112
+ raise 'Ws_path can\'t be nil' unless ws_path
113
+ raise 'Ws_path must be a String' unless ws_path.is_a?(String)
101
114
  @ws_path = ws_path
115
+ raise 'Log can\'t be nil' unless log
102
116
  @log = log
117
+ raise 'Chain can\'t be nil' unless chain
118
+ raise 'Chain must be an Integer' unless chain.is_a?(Integer)
119
+ raise 'Chain must be a positive Integer' unless chain.positive?
103
120
  @chain = chain
104
121
  @proxy = proxy
105
122
  @mutex = Mutex.new
@@ -110,9 +127,11 @@ class ERC20::Wallet
110
127
  # @param [String] hex Public key, in hex, starting from '0x'
111
128
  # @return [Integer] Balance, in tokens
112
129
  def balance(hex)
130
+ raise 'Address can\'t be nil' unless hex
131
+ raise 'Address must be a String' unless hex.is_a?(String)
132
+ raise 'Invalid format of the address' unless /^0x[0-9a-fA-F]{40}$/.match?(hex)
113
133
  func = '70a08231' # balanceOf
114
- padded = "000000000000000000000000#{hex[2..].downcase}"
115
- data = "0x#{func}#{padded}"
134
+ data = "0x#{func}000000000000000000000000#{hex[2..].downcase}"
116
135
  r = jsonrpc.eth_call({ to: @contract, data: data }, 'latest')
117
136
  b = r[2..].to_i(16)
118
137
  @log.debug("Balance of #{hex} is #{b}")
@@ -128,6 +147,23 @@ class ERC20::Wallet
128
147
  # @param [Integer] gas_price How much gas you pay per computation unit
129
148
  # @return [String] Transaction hash
130
149
  def pay(priv, address, amount, gas_limit: nil, gas_price: nil)
150
+ raise 'Private key can\'t be nil' unless priv
151
+ raise 'Private key must be a String' unless priv.is_a?(String)
152
+ raise 'Invalid format of private key' unless /^[0-9a-fA-F]{64}$/.match?(priv)
153
+ raise 'Address can\'t be nil' unless address
154
+ raise 'Address must be a String' unless address.is_a?(String)
155
+ raise 'Invalid format of the address' unless /^0x[0-9a-fA-F]{40}$/.match?(address)
156
+ raise 'Amount can\'t be nil' unless amount
157
+ raise 'Amount must be an Integer' unless amount.is_a?(Integer)
158
+ raise 'Amount must be a positive Integer' unless amount.positive?
159
+ if gas_limit
160
+ raise 'Gas limit must be an Integer' unless gas_limit.is_a?(Integer)
161
+ raise 'Gas limit must be a positive Integer' unless gas_limit.positive?
162
+ end
163
+ if gas_price
164
+ raise 'Gas price must be an Integer' unless gas_price.is_a?(Integer)
165
+ raise 'Gas price must be a positive Integer' unless gas_price.positive?
166
+ end
131
167
  func = 'a9059cbb' # transfer(address,uint256)
132
168
  to_clean = address.downcase.sub(/^0x/, '')
133
169
  to_padded = ('0' * (64 - to_clean.size)) + to_clean
@@ -169,42 +205,47 @@ class ERC20::Wallet
169
205
  # Once we actually start listening, the +active+ array will be updated
170
206
  # with the list of addresses.
171
207
  #
172
- # Both +addresses+ and +active+ must have two methods implemented: +to_a()+
173
- # and +append()+. Only these methods will be called.
208
+ # The +addresses+ must have +to_a()+ implemented.
209
+ # The +active+ must have +append()+ implemented.
210
+ # Only these methods will be called.
174
211
  #
175
212
  # @param [Array<String>] addresses Addresses to monitor
176
213
  # @param [Array] active List of addresses that we are actually listening to
177
214
  # @param [Boolean] raw TRUE if you need to get JSON events as they arrive from Websockets
178
215
  # @param [Integer] delay How many seconds to wait between +eth_subscribe+ calls
179
- def accept(addresses, active = [], raw: false, delay: 1)
216
+ # @param [Integer] subscription_id Unique ID of the subscription
217
+ def accept(addresses, active = [], raw: false, delay: 1, subscription_id: rand(99_999))
218
+ raise 'Addresses can\'t be nil' unless addresses
219
+ raise 'Addresses must respond to .to_a()' unless addresses.respond_to?(:to_a)
220
+ raise 'Active can\'t be nil' unless active
221
+ raise 'Active must respond to .append()' unless active.respond_to?(:append)
222
+ raise 'Amount must be an Integer' unless delay.is_a?(Integer)
223
+ raise 'Amount must be a positive Integer' unless delay.positive?
224
+ raise 'Subscription ID must be an Integer' unless subscription_id.is_a?(Integer)
225
+ raise 'Subscription ID must be a positive Integer' unless subscription_id.positive?
180
226
  EventMachine.run do
181
227
  u = url(http: false)
182
228
  @log.debug("Connecting to #{u.hostname}:#{u.port}...")
183
229
  ws = Faye::WebSocket::Client.new(u.to_s, [], proxy: @proxy ? { origin: @proxy } : {})
184
230
  log = @log
185
231
  contract = @contract
186
- id = rand(99_999)
187
232
  attempt = []
233
+ log_url = "ws#{@ssl ? 's' : ''}://#{u.hostname}:#{u.port}"
188
234
  ws.on(:open) do
189
235
  verbose do
190
- log.debug("Connected to ws://#{u.hostname}:#{u.port}")
236
+ log.debug("Connected to #{log_url}")
191
237
  end
192
238
  end
193
239
  ws.on(:message) do |msg|
194
240
  verbose do
195
- data =
196
- begin
197
- JSON.parse(msg.data)
198
- rescue StandardError
199
- {}
200
- end
241
+ data = to_json(msg)
201
242
  if data['id']
202
243
  before = active.to_a
203
244
  attempt.each do |a|
204
245
  active.append(a) unless before.include?(a)
205
246
  end
206
247
  log.debug(
207
- "Subscribed ##{id} to #{active.to_a.size} addresses: " \
248
+ "Subscribed ##{subscription_id} to #{active.to_a.size} addresses at #{log_url}: " \
208
249
  "#{active.to_a.map { |a| a[0..6] }.join(', ')}"
209
250
  )
210
251
  elsif data['method'] == 'eth_subscription' && data.dig('params', 'result')
@@ -218,7 +259,10 @@ class ERC20::Wallet
218
259
  to: "0x#{event['topics'][2][26..].downcase}",
219
260
  txn: event['transactionHash'].downcase
220
261
  }
221
- log.debug("Payment of #{event[:amount]} tokens arrived from #{event[:from]} to #{event[:to]}")
262
+ log.debug(
263
+ "Payment of #{event[:amount]} tokens arrived" \
264
+ "from #{event[:from]} to #{event[:to]} in #{event[:txn]}"
265
+ )
222
266
  end
223
267
  yield event
224
268
  end
@@ -226,12 +270,12 @@ class ERC20::Wallet
226
270
  end
227
271
  ws.on(:close) do
228
272
  verbose do
229
- log.debug("Disconnected from ws://#{u.hostname}:#{u.port}")
273
+ log.debug("Disconnected from #{log_url}")
230
274
  end
231
275
  end
232
276
  ws.on(:error) do |e|
233
277
  verbose do
234
- log.debug("Error at #{u.hostname}: #{e.message}")
278
+ log.debug("Error at #{log_url}: #{e.message}")
235
279
  end
236
280
  end
237
281
  EventMachine.add_periodic_timer(delay) do
@@ -240,7 +284,7 @@ class ERC20::Wallet
240
284
  ws.send(
241
285
  {
242
286
  jsonrpc: '2.0',
243
- id:,
287
+ id: subscription_id,
244
288
  method: 'eth_subscribe',
245
289
  params: [
246
290
  'logs',
@@ -256,7 +300,7 @@ class ERC20::Wallet
256
300
  }.to_json
257
301
  )
258
302
  log.debug(
259
- "Requested to subscribe ##{id} to #{addresses.to_a.size} addresses: " \
303
+ "Requested to subscribe ##{subscription_id} to #{addresses.to_a.size} addresses at #{log_url}: " \
260
304
  "#{addresses.to_a.map { |a| a[0..6] }.join(', ')}"
261
305
  )
262
306
  end
@@ -265,6 +309,12 @@ class ERC20::Wallet
265
309
 
266
310
  private
267
311
 
312
+ def to_json(msg)
313
+ JSON.parse(msg.data)
314
+ rescue StandardError
315
+ {}
316
+ end
317
+
268
318
  def verbose
269
319
  yield
270
320
  rescue StandardError => e
@@ -33,6 +33,10 @@ require 'typhoeus'
33
33
  require_relative '../../lib/erc20/fake_wallet'
34
34
  require_relative '../test__helper'
35
35
 
36
+ k = Eth::Key.new
37
+ puts k.private_hex
38
+ puts k.address
39
+
36
40
  # Test.
37
41
  # Author:: Yegor Bugayenko (yegor256@gmail.com)
38
42
  # Copyright:: Copyright (c) 2025 Yegor Bugayenko
@@ -38,9 +38,9 @@ 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 a ~$27 USDT. I won't
41
+ # At this address, in Etherium mainnet, there are $8 USDT. I won't
42
42
  # move them anyway, that's why tests can use this address forever.
43
- STABLE = '0xEB2fE8872A6f1eDb70a2632EA1f869AB131532f6'
43
+ STABLE = '0x7232148927F8a580053792f44D4d59d40Fd00ABD'
44
44
 
45
45
  # One guy private hex.
46
46
  JEFF = '81a9b2114d53731ecc84b261ef6c0387dde34d5907fe7b441240cc21d61bf80a'
@@ -51,7 +51,7 @@ 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(27_258_889, b)
54
+ assert_equal(8_000_000, b)
55
55
  end
56
56
 
57
57
  def test_checks_balance_of_absent_address
@@ -253,11 +253,21 @@ class TestWallet < Minitest::Test
253
253
  host: 'mainnet.infura.io', http_path: "/v3/#{env('INFURA_KEY')}",
254
254
  proxy:, log: loog
255
255
  )
256
- assert_equal(27_258_889, w.balance(STABLE))
256
+ assert_equal(8_000_000, w.balance(STABLE))
257
257
  end
258
258
  end
259
259
  end
260
260
 
261
+ def test_pays_on_mainnet
262
+ skip('This is live, must be run manually')
263
+ w = mainnet
264
+ print 'Enter Etherium ERC20 private key (64 chars): '
265
+ priv = gets.chomp
266
+ to = '0xEB2fE8872A6f1eDb70a2632EA1f869AB131532f6'
267
+ txn = w.pay(priv, to, 1_990_000)
268
+ assert_equal(66, txn.length)
269
+ end
270
+
261
271
  private
262
272
 
263
273
  def env(var)
data/test/test__helper.rb CHANGED
@@ -64,12 +64,13 @@ class Minitest::Test
64
64
  ENV['RAKE'] ? Loog::ERRORS : Loog::VERBOSE
65
65
  end
66
66
 
67
- def wait_for
67
+ def wait_for(seconds = 30)
68
68
  start = Time.now
69
69
  loop do
70
70
  sleep(0.1)
71
71
  break if yield
72
- raise 'timeout' if Time.now - start > 60
72
+ passed = Time.now - start
73
+ raise "Giving up after #{passed} seconds of waiting" if passed > seconds
73
74
  rescue Errno::ECONNREFUSED
74
75
  retry
75
76
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: erc20
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.0.12
4
+ version: 0.0.14
5
5
  platform: ruby
6
6
  authors:
7
7
  - Yegor Bugayenko
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2025-02-14 00:00:00.000000000 Z
11
+ date: 2025-02-16 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: eth