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 +4 -4
- data/.rubocop.yml +2 -2
- data/Gemfile +3 -3
- data/Gemfile.lock +19 -14
- data/lib/erc20/erc20.rb +1 -1
- data/lib/erc20/wallet.rb +69 -19
- data/test/erc20/test_fake_wallet.rb +4 -0
- data/test/erc20/test_wallet.rb +14 -4
- data/test/test__helper.rb +3 -2
- metadata +2 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: ac1bd8d3ffe6bf19ceb44174514dca9abb57746110c864821de775b2f11f955a
|
4
|
+
data.tar.gz: 7cc7b1d4fe20626629ea99ca5bb6cbb02686c872e36106c4c05fab846ad9b24b
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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
|
-
|
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.
|
38
|
-
gem 'rubocop-minitest', '0.
|
39
|
-
gem 'rubocop-performance', '1.
|
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.
|
205
|
+
rubocop (1.72.1)
|
205
206
|
json (~> 2.3)
|
206
|
-
language_server-protocol (
|
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.
|
217
|
-
|
218
|
-
rubocop
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
|
223
|
-
rubocop (
|
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.
|
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.
|
286
|
-
rubocop-minitest (= 0.
|
287
|
-
rubocop-performance (= 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
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
|
-
|
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
|
-
#
|
173
|
-
#
|
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
|
-
|
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
|
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 ##{
|
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(
|
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
|
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 #{
|
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 ##{
|
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
|
data/test/erc20/test_wallet.rb
CHANGED
@@ -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
|
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 = '
|
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(
|
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(
|
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
|
-
|
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.
|
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-
|
11
|
+
date: 2025-02-16 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: eth
|