cns 0.9.6 → 0.9.8
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/Gemfile.lock +4 -1
- data/cns.gemspec +1 -0
- data/lib/cns/apibc.rb +8 -3
- data/lib/cns/apice.rb +7 -1
- data/lib/cns/bitcoinde.rb +6 -2
- data/lib/cns/etherscan.rb +99 -62
- data/lib/cns/greymass.rb +46 -39
- data/lib/cns/kraken.rb +16 -2
- data/lib/cns/version.rb +1 -1
- metadata +16 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: f44706093bb37569dad944e52da3becb0d1bb8ab34e48c9dd30c2f2cd6599f3c
|
4
|
+
data.tar.gz: 1cb9e471623d4abbf526a77278e5ff27614e52093d3a79a56229a9c9bf39783f
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 99c1e87bc81a61b9c8e572807b133daf02e54b42bbdea7b2d1997711fc4a98984eae1320ebc6764ce15cd586dee29f88716fa49bc07bb91f136f9a4dd4bf3d55
|
7
|
+
data.tar.gz: 0ed7c0a9c0b40fcb5d18cf7271a5208ddd62e2e0643c0ae2134d74538fb6993c6756ba962e7de754751feae71873e2cb22644c74d5e6b999f78bda6336714e94
|
data/Gemfile.lock
CHANGED
@@ -1,9 +1,10 @@
|
|
1
1
|
PATH
|
2
2
|
remote: .
|
3
3
|
specs:
|
4
|
-
cns (0.9.
|
4
|
+
cns (0.9.8)
|
5
5
|
curb
|
6
6
|
faraday
|
7
|
+
faraday-retry
|
7
8
|
google-cloud-bigquery
|
8
9
|
thor
|
9
10
|
|
@@ -56,6 +57,8 @@ GEM
|
|
56
57
|
logger
|
57
58
|
faraday-net_http (3.4.0)
|
58
59
|
net-http (>= 0.5.0)
|
60
|
+
faraday-retry (2.2.1)
|
61
|
+
faraday (~> 2.0)
|
59
62
|
google-apis-bigquery_v2 (0.84.0)
|
60
63
|
google-apis-core (>= 0.15.0, < 2.a)
|
61
64
|
google-apis-core (0.16.0)
|
data/cns.gemspec
CHANGED
@@ -36,6 +36,7 @@ Gem::Specification.new do |spec|
|
|
36
36
|
|
37
37
|
spec.add_dependency('curb')
|
38
38
|
spec.add_dependency('faraday')
|
39
|
+
spec.add_dependency('faraday-retry')
|
39
40
|
spec.add_dependency('google-cloud-bigquery')
|
40
41
|
spec.add_dependency('thor')
|
41
42
|
spec.metadata['rubygems_mfa_required'] = 'true'
|
data/lib/cns/apibc.rb
CHANGED
@@ -1,6 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
require('faraday')
|
4
|
+
require 'faraday/retry'
|
4
5
|
require('json')
|
5
6
|
|
6
7
|
# @author Hernani Rodrigues Vaz
|
@@ -22,8 +23,11 @@ module Cns
|
|
22
23
|
def account_es(addresses)
|
23
24
|
return [] if addresses.empty?
|
24
25
|
|
25
|
-
|
26
|
-
|
26
|
+
# Batch addresses into groups of 20 (Etherscan limit) and fetch balances
|
27
|
+
addresses.each_slice(20).flat_map do |b|
|
28
|
+
res = es_req('balancemulti', b.join(','), 1, tag: 'latest')
|
29
|
+
res[:status] == '1' ? res[:result] || [] : []
|
30
|
+
end
|
27
31
|
end
|
28
32
|
|
29
33
|
# Get normal transactions for ETH address
|
@@ -149,7 +153,7 @@ module Cns
|
|
149
153
|
{}
|
150
154
|
end
|
151
155
|
|
152
|
-
# Create a Faraday connection with JSON configuration
|
156
|
+
# Create a Faraday connection with JSON configuration and retry logic
|
153
157
|
# @param [String] url Base URL for the API
|
154
158
|
# @return [Faraday::Connection] Configured Faraday connection
|
155
159
|
def connection(url)
|
@@ -158,6 +162,7 @@ module Cns
|
|
158
162
|
c.headers = {accept: 'application/json', user_agent: 'blockchain-api-client'}
|
159
163
|
c.options.timeout = 30
|
160
164
|
c.options.open_timeout = 10
|
165
|
+
c.use(Faraday::Retry::Middleware, max: 3, interval: 1)
|
161
166
|
c.adapter(Faraday.default_adapter)
|
162
167
|
end
|
163
168
|
end
|
data/lib/cns/apice.rb
CHANGED
@@ -106,7 +106,8 @@ module Cns
|
|
106
106
|
ary = []
|
107
107
|
ofs = 0
|
108
108
|
loop do
|
109
|
-
|
109
|
+
# Rate limiting for page requests (2s in Kraken)
|
110
|
+
sleep(@lpag - Time.now + 2) if @lpag && Time.now - @lpag < 2
|
110
111
|
ops = {nonce: nnc, ofs: ofs}
|
111
112
|
run_curl(@curl, "#{API[:us]}/#{uri}", method: 'POST', post_data: ops, headers: hus(uri, ops))
|
112
113
|
bth = parse_json(@curl).fetch(:result, {}).fetch(key, []).map { |k, v| us_unif(k, v) }
|
@@ -114,6 +115,7 @@ module Cns
|
|
114
115
|
|
115
116
|
ary.concat(bth)
|
116
117
|
ofs += bth.size
|
118
|
+
@lpag = Time.now
|
117
119
|
end
|
118
120
|
ary
|
119
121
|
end
|
@@ -200,6 +202,8 @@ module Cns
|
|
200
202
|
md5 = ['GET', qde, @deky, non, Digest::MD5.hexdigest('')].join('#')
|
201
203
|
mac = OpenSSL::HMAC.hexdigest('sha256', @desc, md5)
|
202
204
|
{'X-API-KEY' => @deky, 'X-API-NONCE' => non.to_s, 'X-API-SIGNATURE' => mac}
|
205
|
+
rescue OpenSSL::HMACError => e
|
206
|
+
raise("HMAC bitcoinde generation failed: #{e.message}")
|
203
207
|
end
|
204
208
|
|
205
209
|
# Generate headers for Kraken HTTP requests
|
@@ -211,6 +215,8 @@ module Cns
|
|
211
215
|
sha = ['/0/private/', qus, Digest::SHA256.digest("#{ops[:nonce]}#{URI.encode_www_form(ops)}")].join
|
212
216
|
mac = OpenSSL::HMAC.digest('sha512', Base64.decode64(@ussc), sha)
|
213
217
|
{'api-key' => @usky, 'api-sign' => Base64.strict_encode64(mac)}
|
218
|
+
rescue OpenSSL::HMACError => e
|
219
|
+
raise("HMAC kraken generation failed: #{e.message}")
|
214
220
|
end
|
215
221
|
end
|
216
222
|
end
|
data/lib/cns/bitcoinde.rb
CHANGED
@@ -16,9 +16,7 @@ module Cns
|
|
16
16
|
# @option pop [Hash] :h ({}) configuracao dias ajuste reposicionamento temporal
|
17
17
|
# @option pop [Boolean] :v (false) mostra dados transacoes trades & ledger?
|
18
18
|
# @option pop [Boolean] :t (false) mostra transacoes todas ou somente novas?
|
19
|
-
# @return [Bitcoinde] API bitcoinde - obter saldos & transacoes trades e ledger
|
20
19
|
def initialize(dad, pop)
|
21
|
-
@api = Apice.new
|
22
20
|
@bqd = dad
|
23
21
|
@ops = pop.transform_keys(&:to_sym)
|
24
22
|
end
|
@@ -153,6 +151,12 @@ module Cns
|
|
153
151
|
hlx.map { |t| pdes(:time, t) }
|
154
152
|
end
|
155
153
|
|
154
|
+
# Lazy Bitcoinde API Initialization
|
155
|
+
# @return [Bitcoinde] API - obter saldos & transacoes trades e ledger
|
156
|
+
def api
|
157
|
+
@api ||= Apice.new
|
158
|
+
end
|
159
|
+
|
156
160
|
# @return [Hash] dados exchange bitcoinde - saldos & trades & deposits & withdrawals
|
157
161
|
def ded
|
158
162
|
@ded ||= {sl: pdea(api.account_de), tt: pdet(api.trades_de), tl: pdel(api.deposits_de + api.withdrawals_de)}
|
data/lib/cns/etherscan.rb
CHANGED
@@ -53,9 +53,7 @@ module Cns
|
|
53
53
|
# @param [Thor::CoreExt::HashWithIndifferentAccess] pop opcoes trabalho
|
54
54
|
# @option pop [Hash] :h ({}) configuracao dias ajuste reposicionamento temporal
|
55
55
|
# @option pop [Boolean] :v (false) mostra dados transacoes
|
56
|
-
# @return [Etherscan] API etherscan - processar transacoes
|
57
56
|
def initialize(dad, pop)
|
58
|
-
@api = Apibc.new
|
59
57
|
@bqd = dad
|
60
58
|
@ops = pop.transform_keys(&:to_sym)
|
61
59
|
end
|
@@ -103,42 +101,45 @@ module Cns
|
|
103
101
|
end
|
104
102
|
end
|
105
103
|
|
104
|
+
# Format simple wallet summary
|
106
105
|
# @param [Hash] hjn dados juntos bigquery & etherscan
|
107
106
|
# @return [String] texto formatado duma carteira
|
108
107
|
def focs(hjn)
|
109
108
|
format(
|
110
|
-
'%<
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
|
115
|
-
|
109
|
+
'%<id>-6.6s %<address>-42.42s %<etherscan_value>13.6f %<bigquery_value>13.6f %<status>-3s',
|
110
|
+
id: hjn[:id],
|
111
|
+
address: hjn[:ax],
|
112
|
+
etherscan_value: hjn[:es],
|
113
|
+
bigquery_value: hjn[:bs],
|
114
|
+
status: ok?(hjn) ? 'OK' : 'NOK'
|
116
115
|
)
|
117
116
|
end
|
118
117
|
|
118
|
+
# Format detailed wallet summary with counters
|
119
119
|
# @param (see focs)
|
120
120
|
# @return [String] texto formatado duma carteira (com contadores)
|
121
121
|
def foct(hjn)
|
122
122
|
format(
|
123
|
-
'%<
|
124
|
-
|
125
|
-
|
126
|
-
|
127
|
-
|
128
|
-
|
129
|
-
|
130
|
-
|
131
|
-
|
132
|
-
|
133
|
-
|
134
|
-
|
135
|
-
|
136
|
-
|
137
|
-
|
123
|
+
'%<id>-6.6s %<ax>-10.10s %<es>11.4f %<et>3i %<ei>2i %<ep>2i %<ek>2i %<ew>4i %<bs>11.4f %<bt>3i %<bi>2i %<bp>2i %<bk>2i %<bw>4i %<ok>-3s',
|
124
|
+
id: hjn[:id],
|
125
|
+
ax: foe1(hjn[:ax], 10),
|
126
|
+
es: hjn[:es],
|
127
|
+
et: hjn[:et].count,
|
128
|
+
ei: hjn[:ei].count,
|
129
|
+
ep: hjn[:ep].count,
|
130
|
+
ek: hjn[:ek].count,
|
131
|
+
ew: hjn[:ew].count,
|
132
|
+
bs: hjn[:bs],
|
133
|
+
bt: hjn[:bt].count,
|
134
|
+
bi: hjn[:bi].count,
|
135
|
+
bp: hjn[:bp].count,
|
136
|
+
bk: hjn[:bk].count,
|
137
|
+
bw: hjn[:bw].count,
|
138
138
|
ok: ok?(hjn) ? 'OK' : 'NOK'
|
139
139
|
)
|
140
140
|
end
|
141
141
|
|
142
|
+
# Check if wallet has new transactions
|
142
143
|
# @param (see focs)
|
143
144
|
# @return [Boolean] check saldo & contadores ipwtk
|
144
145
|
def ok?(hjn)
|
@@ -187,50 +188,68 @@ module Cns
|
|
187
188
|
"#{ndd[0, ini]}..#{ndd[-inf - ini..]}"
|
188
189
|
end
|
189
190
|
|
190
|
-
#
|
191
|
+
# Format normal(t)/(i)nternal transaction
|
192
|
+
# @param [Hash] htx transacao etherscan
|
191
193
|
# @return [String] texto formatado
|
192
194
|
def foti(htx)
|
193
195
|
format(
|
194
|
-
'%<
|
195
|
-
|
196
|
-
|
196
|
+
'%<hash>-29.29s %<from>-15.15s %<to>-15.15s %<date>10.10s %<value>7.3f',
|
197
|
+
hash: foe1(htx[:hash], 29),
|
198
|
+
from: foe2(htx[:from], 15),
|
197
199
|
to: foe2(htx[:to], 15),
|
198
|
-
|
199
|
-
|
200
|
+
date: htx[:timeStamp].strftime('%F'),
|
201
|
+
value: htx[:value] / (10**18)
|
200
202
|
)
|
201
203
|
end
|
202
204
|
|
203
|
-
#
|
205
|
+
# Format to(k)en transaction
|
206
|
+
# @param [Hash] hkx transacao etherscan
|
204
207
|
# @return [String] texto formatado
|
205
208
|
def fok(hkx)
|
206
209
|
format(
|
207
|
-
'%<
|
208
|
-
|
209
|
-
|
210
|
+
'%<hash>-20.20s %<from>-15.15s %<to>-15.15s %<date>10.10s %<value>10.3f %<symbol>-5.5s',
|
211
|
+
hash: foe1(hkx[:hash], 20),
|
212
|
+
from: foe2(hkx[:from], 15),
|
210
213
|
to: foe2(hkx[:to], 15),
|
211
|
-
|
212
|
-
|
213
|
-
|
214
|
+
date: hkx[:timeStamp].strftime('%F'),
|
215
|
+
value: hkx[:value] / (10**18),
|
216
|
+
symbol: hkx[:tokenSymbol]
|
214
217
|
)
|
215
218
|
end
|
216
219
|
|
217
|
-
#
|
220
|
+
# Format (p)roduced block transaction
|
221
|
+
# @param [Hash] hpx transacao etherscan
|
218
222
|
# @return [String] texto formatado
|
219
223
|
def fop(hpx)
|
220
|
-
format(
|
224
|
+
format(
|
225
|
+
'%<block_number>9i %<address>-41.41s %<date>10.10s %<reward>17.6f',
|
226
|
+
block_number: hpx[:blockNumber],
|
227
|
+
address: foe2(hpx[:iax], 41),
|
228
|
+
date: hpx[:timeStamp].strftime('%F'),
|
229
|
+
reward: hpx[:blockReward] / (10**18)
|
230
|
+
)
|
221
231
|
end
|
222
232
|
|
223
|
-
#
|
224
|
-
# @
|
233
|
+
# Format (w)ithdrawal transaction
|
234
|
+
# @param [Hash] hwx transacao etherscan
|
235
|
+
# @return [String] texto formatado
|
225
236
|
def fow(hwx)
|
226
|
-
format(
|
237
|
+
format(
|
238
|
+
'%<index>10i %<validator>9i %<date>10.10s %<amount>10.6f',
|
239
|
+
index: hwx[:withdrawalIndex],
|
240
|
+
validator: hwx[:validatorIndex],
|
241
|
+
date: hwx[:timeStamp].strftime('%F'),
|
242
|
+
amount: hwx[:amount] / (10**9)
|
243
|
+
)
|
227
244
|
end
|
228
245
|
|
246
|
+
# Determine if all transactions should be shown
|
229
247
|
# @return [Boolean] mostra todas/novas transacoes
|
230
248
|
def show_all?
|
231
249
|
ops[:t] || false
|
232
250
|
end
|
233
251
|
|
252
|
+
# Process timestamp
|
234
253
|
# @param [Hash] htx transacao
|
235
254
|
# @return [Hash] transaccao filtrada
|
236
255
|
def pess(htx)
|
@@ -240,27 +259,31 @@ module Cns
|
|
240
259
|
htx.merge(srx: 0, timeStamp: Time.at(0))
|
241
260
|
end
|
242
261
|
|
262
|
+
# Filter normal(t)/(i)nternal/to(k)en transactions
|
243
263
|
# @param add (see foe1)
|
244
|
-
# @param [Array<Hash>] ary lista transacoes
|
264
|
+
# @param [Array<Hash>] ary lista transacoes
|
245
265
|
# @return [Array<Hash>] lista transacoes filtrada
|
246
266
|
def ftik(add, ary)
|
247
267
|
ary.map { |o| pess(o).merge(itx: o[:hash].to_s, iax: add, value: o[:value].to_d) }
|
248
268
|
end
|
249
269
|
|
270
|
+
# Filter (p)roduced blocks transactions
|
250
271
|
# @param add (see foe1)
|
251
|
-
# @param [Array<Hash>] ary lista transacoes
|
272
|
+
# @param [Array<Hash>] ary lista transacoes
|
252
273
|
# @return [Array<Hash>] lista transacoes filtrada
|
253
274
|
def fppp(add, ary)
|
254
275
|
ary.map { |o| o.merge(itx: o[:blockNumber].to_i, iax: add, blockReward: o[:blockReward].to_d, timeStamp: Time.at(o[:timeStamp].to_i)) }
|
255
276
|
end
|
256
277
|
|
278
|
+
# Filter (w)ithdrawals transactions
|
257
279
|
# @param add (see foe1)
|
258
|
-
# @param [Array<Hash>] ary lista transacoes
|
280
|
+
# @param [Array<Hash>] ary lista transacoes
|
259
281
|
# @return [Array<Hash>] lista transacoes filtrada
|
260
282
|
def fwww(add, ary)
|
261
283
|
ary.map { |o| o.merge(itx: o[:withdrawalIndex].to_i, iax: add, amount: o[:amount].to_d, timeStamp: Time.at(o[:timestamp].to_i)) }
|
262
284
|
end
|
263
285
|
|
286
|
+
# Fetch Etherscan data for an account
|
264
287
|
# @param [Hash] aes account etherscan
|
265
288
|
# @return [Hash] dados etherscan - address, saldo & transacoes
|
266
289
|
def bses(aes)
|
@@ -276,6 +299,7 @@ module Cns
|
|
276
299
|
}
|
277
300
|
end
|
278
301
|
|
302
|
+
# Combine BigQuery and Etherscan data
|
279
303
|
# @param [Hash] wbq wallet bigquery
|
280
304
|
# @param [Hash] hes dados etherscan - address, saldo & transacoes
|
281
305
|
# @return [Hash] dados juntos bigquery & etherscan
|
@@ -299,19 +323,27 @@ module Cns
|
|
299
323
|
}
|
300
324
|
end
|
301
325
|
|
326
|
+
# Lazy Etherscan API Initialization
|
327
|
+
# @return [Apibc] API instance
|
328
|
+
def api
|
329
|
+
@api ||= Apibc.new
|
330
|
+
end
|
331
|
+
|
302
332
|
# @return [Array<String>] lista enderecos
|
303
333
|
def lax
|
304
334
|
@lax ||= bqd[:wb].map { |o| o[:ax] }
|
305
335
|
end
|
306
336
|
|
307
|
-
#
|
337
|
+
# Fetch all Etherscan data
|
338
|
+
# @return [Hash] saldos & transacoes, indexed by address
|
308
339
|
def esd
|
309
|
-
@esd ||= api.account_es(lax).map { |o| bses(o) }
|
340
|
+
@esd ||= api.account_es(lax).map { |o| bses(o) }.each_with_object({}) { |h, a| a[h[:ax]] = h }
|
310
341
|
end
|
311
342
|
|
343
|
+
# Fetch combined data
|
312
344
|
# @return [Array<Hash>] todos os dados juntos bigquery & etherscan
|
313
345
|
def dados
|
314
|
-
@dados ||= bqd[:wb].map { |b| bqes(b, esd
|
346
|
+
@dados ||= bqd[:wb].map { |b| bqes(b, esd[b[:ax]]) }
|
315
347
|
end
|
316
348
|
|
317
349
|
# @return [Array<Integer>] indices transacoes bigquery
|
@@ -341,52 +373,57 @@ module Cns
|
|
341
373
|
|
342
374
|
# @return [Array<Integer>] indices transacoes novas (etherscan - bigquery)
|
343
375
|
def idt
|
344
|
-
@idt ||= esd.map { |o| o[:tx].map { |i| i[:itx] } }.flatten - bqidt
|
376
|
+
@idt ||= esd.values.map { |o| o[:tx].map { |i| i[:itx] } }.flatten - bqidt
|
345
377
|
end
|
346
378
|
|
347
379
|
# @return [Array<Integer>] indices transacoes novas (etherscan - bigquery)
|
348
380
|
def idi
|
349
|
-
@idi ||= esd.map { |o| o[:ix].map { |i| i[:itx] } }.flatten - bqidi
|
381
|
+
@idi ||= esd.values.map { |o| o[:ix].map { |i| i[:itx] } }.flatten - bqidi
|
350
382
|
end
|
351
383
|
|
352
384
|
# @return [Array<Integer>] indices transacoes novas (etherscan - bigquery)
|
353
385
|
def idp
|
354
|
-
@idp ||= esd.map { |o| o[:px].map { |i| i[:itx] } }.flatten - bqidp
|
386
|
+
@idp ||= esd.values.map { |o| o[:px].map { |i| i[:itx] } }.flatten - bqidp
|
355
387
|
end
|
356
388
|
|
357
389
|
# @return [Array<Integer>] indices transacoes novas (etherscan - bigquery)
|
358
390
|
def idw
|
359
|
-
@idw ||= esd.map { |o| o[:wx].map { |i| i[:itx] } }.flatten - bqidw
|
391
|
+
@idw ||= esd.values.map { |o| o[:wx].map { |i| i[:itx] } }.flatten - bqidw
|
360
392
|
end
|
361
393
|
|
362
394
|
# @return [Array<Integer>] indices transacoes novas (etherscan - bigquery)
|
363
395
|
def idk
|
364
|
-
@idk ||= esd.map { |o| o[:kx].map { |i| i[:itx] } }.flatten - bqidk
|
396
|
+
@idk ||= esd.values.map { |o| o[:kx].map { |i| i[:itx] } }.flatten - bqidk
|
365
397
|
end
|
366
398
|
|
367
|
-
#
|
399
|
+
# Get new normal transactions
|
400
|
+
# @return [Array<Hash>] List of new transactions
|
368
401
|
def novnetht
|
369
|
-
@novnetht ||= esd.map { |o| o[:tx].select { |t| idt.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
|
402
|
+
@novnetht ||= esd.values.map { |o| o[:tx].select { |t| idt.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
|
370
403
|
end
|
371
404
|
|
372
|
-
#
|
405
|
+
# Get new internal transactions
|
406
|
+
# @return [Array<Hash>] List of new transactions
|
373
407
|
def novnethi
|
374
|
-
@novnethi ||= esd.map { |o| o[:ix].select { |t| idi.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
|
408
|
+
@novnethi ||= esd.values.map { |o| o[:ix].select { |t| idi.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
|
375
409
|
end
|
376
410
|
|
377
|
-
#
|
411
|
+
# Get new produced block transactions
|
412
|
+
# @return [Array<Hash>] List of new transactions
|
378
413
|
def novnethp
|
379
|
-
@novnethp ||= esd.map { |o| o[:px].select { |t| idp.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
|
414
|
+
@novnethp ||= esd.values.map { |o| o[:px].select { |t| idp.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
|
380
415
|
end
|
381
416
|
|
382
|
-
#
|
417
|
+
# Get new withdrawal transactions
|
418
|
+
# @return [Array<Hash>] List of new transactions
|
383
419
|
def novnethw
|
384
|
-
@novnethw ||= esd.map { |o| o[:wx].select { |t| idw.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
|
420
|
+
@novnethw ||= esd.values.map { |o| o[:wx].select { |t| idw.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
|
385
421
|
end
|
386
422
|
|
387
|
-
#
|
423
|
+
# Get new token transactions
|
424
|
+
# @return [Array<Hash>] List of new transactions
|
388
425
|
def novnethk
|
389
|
-
@novnethk ||= esd.map { |o| o[:kx].select { |t| idk.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
|
426
|
+
@novnethk ||= esd.values.map { |o| o[:kx].select { |t| idk.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
|
390
427
|
end
|
391
428
|
end
|
392
429
|
end
|
data/lib/cns/greymass.rb
CHANGED
@@ -11,32 +11,24 @@ module Cns
|
|
11
11
|
# @return [Thor::CoreExt::HashWithIndifferentAccess] opcoes trabalho
|
12
12
|
attr_reader :api, :bqd, :ops
|
13
13
|
|
14
|
-
TT = {
|
15
|
-
new: :novneost,
|
16
|
-
format: :fol,
|
17
|
-
header: "\nsequence num from to accao data valor moeda",
|
18
|
-
sork: :itx,
|
19
|
-
adjk: :itx
|
20
|
-
}.freeze
|
14
|
+
TT = {sork: :itx, adjk: :itx}.freeze
|
21
15
|
|
22
16
|
# @param [Hash] dad todos os dados bigquery
|
23
17
|
# @param [Thor::CoreExt::HashWithIndifferentAccess] pop opcoes trabalho
|
24
18
|
# @option pop [Hash] :h ({}) configuracao dias ajuste reposicionamento temporal
|
25
19
|
# @option pop [Boolean] :v (false) mostra dados transacoes?
|
26
20
|
# @option pop [Boolean] :t (false) mostra transacoes todas ou somente novas?
|
27
|
-
# @return [Greymass] API greymass - processar transacoes
|
28
21
|
def initialize(dad, pop)
|
29
|
-
@api = Apibc.new
|
30
22
|
@bqd = dad
|
31
23
|
@ops = pop.transform_keys(&:to_sym)
|
32
24
|
end
|
33
25
|
|
34
|
-
#
|
26
|
+
# Display summary of wallets, transactions, and adjustment days configuration
|
35
27
|
def mresumo
|
36
28
|
return unless dados.any?
|
37
29
|
|
38
30
|
puts("\naddress greymass ntx bigquery ntx")
|
39
|
-
dados.each { |
|
31
|
+
dados.each { |e| puts(foct(e)) }
|
40
32
|
mtransacoes_novas
|
41
33
|
mconfiguracao_ajuste_dias
|
42
34
|
end
|
@@ -45,61 +37,64 @@ module Cns
|
|
45
37
|
|
46
38
|
# mosta transacoes novas
|
47
39
|
def mtransacoes_novas
|
48
|
-
|
49
|
-
return unless ops[:v] && ntx.any?
|
40
|
+
return unless ops[:v] && novneost.any?
|
50
41
|
|
51
|
-
puts(
|
52
|
-
|
42
|
+
puts("\nsequence num from to accao data valor moeda")
|
43
|
+
novneost.sort_by { |s| -s[TT[:sork]] }.each { |t| puts(fol(t)) }
|
53
44
|
end
|
54
45
|
|
55
46
|
# mostra configuration text for adjusting days
|
56
47
|
def mconfiguracao_ajuste_dias
|
57
|
-
|
58
|
-
return unless ntx.any?
|
48
|
+
return unless novneost.any?
|
59
49
|
|
60
|
-
puts("\nstring ajuste dias\n-h=#{
|
50
|
+
puts("\nstring ajuste dias\n-h=#{novneost.sort_by { |s| -s[TT[:sork]] }.map { |t| "#{t[TT[:adjk]]}:0" }.join(' ')}")
|
61
51
|
end
|
62
52
|
|
53
|
+
# Format wallet summary text
|
63
54
|
# @param [Hash] hjn dados juntos bigquery & greymass
|
64
55
|
# @return [String] texto formatado duma carteira
|
65
56
|
def foct(hjn)
|
66
57
|
format(
|
67
|
-
'%<
|
68
|
-
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
73
|
-
|
58
|
+
'%<address>-12.12s %<greymass_value>14.4f %<greymass_tx_count>4i %<bigquery_value>14.4f %<bigquery_tx_count>4i %<status>-3s',
|
59
|
+
address: hjn[:ax],
|
60
|
+
greymass_value: hjn[:es],
|
61
|
+
greymass_tx_count: hjn[:et].count,
|
62
|
+
bigquery_value: hjn[:bs],
|
63
|
+
bigquery_tx_count: hjn[:bt].count,
|
64
|
+
status: ok?(hjn) ? 'OK' : 'NOK'
|
74
65
|
)
|
75
66
|
end
|
76
67
|
|
68
|
+
# Check if wallet has new transactions
|
77
69
|
# @param (see foct)
|
78
70
|
# @return [Boolean] carteira tem transacoes novas(sim=NOK, nao=OK)?
|
79
71
|
def ok?(hjn)
|
80
72
|
hjn[:bs].round(6) == hjn[:es].round(6) && hjn[:bt].count == hjn[:et].count
|
81
73
|
end
|
82
74
|
|
75
|
+
# Format transaction text
|
83
76
|
# @param [Hash] hlx ledger greymass
|
84
77
|
# @return [String] texto formatado
|
85
78
|
def fol(hlx)
|
86
79
|
format(
|
87
|
-
'%<
|
88
|
-
|
89
|
-
|
90
|
-
vl: hlx[:quantity],
|
91
|
-
bn: hlx[:itx],
|
80
|
+
'%<sequence>12i %<from>-12.12s %<to>-12.12s %<action>-10.10s %<date>10.10s %<value>12.4f %<symbol>-6.6s',
|
81
|
+
sequence: hlx[:itx],
|
82
|
+
from: hlx[:from],
|
92
83
|
to: hlx[:to],
|
93
|
-
|
94
|
-
|
84
|
+
action: hlx[:name],
|
85
|
+
date: hlx[:block_time].strftime('%F'),
|
86
|
+
value: hlx[:quantity],
|
87
|
+
symbol: hlx[:moe]
|
95
88
|
)
|
96
89
|
end
|
97
90
|
|
91
|
+
# Determine if all transactions should be shown
|
98
92
|
# @return [Boolean] mostra todas/novas transacoes
|
99
93
|
def show_all?
|
100
94
|
ops[:t] || false
|
101
95
|
end
|
102
96
|
|
97
|
+
# Fetch EOS account resources
|
103
98
|
# @param [String] add EOS account name
|
104
99
|
# @return [Array<BigDecimal>] lista recursos - liquido, net, spu
|
105
100
|
def peosa(add)
|
@@ -108,6 +103,7 @@ module Cns
|
|
108
103
|
[hac[:core_liquid_balance]&.to_d || 0.to_d, htr[:net_weight]&.to_d || 0.to_d, htr[:cpu_weight]&.to_d || 0.to_d]
|
109
104
|
end
|
110
105
|
|
106
|
+
# Process and filter EOS transactions
|
111
107
|
# @param add (see peosa)
|
112
108
|
# @param [Array<Hash>] ary lista transacoes
|
113
109
|
# @return [Array<Hash>] lista transacoes filtrada
|
@@ -131,6 +127,7 @@ module Cns
|
|
131
127
|
end
|
132
128
|
end
|
133
129
|
|
130
|
+
# Fetch Greymass data for a wallet
|
134
131
|
# @param [Hash] wbq wallet bigquery
|
135
132
|
# @return [Hash] dados greymass - address, saldo & transacoes
|
136
133
|
def bsgm(wbq)
|
@@ -138,6 +135,7 @@ module Cns
|
|
138
135
|
{ax: xbq, sl: peosa(xbq).reduce(:+), tx: peost(xbq, api.ledger_gm(xbq))}
|
139
136
|
end
|
140
137
|
|
138
|
+
# Combine BigQuery and Greymass data
|
141
139
|
# @param wbq (see bsgm)
|
142
140
|
# @param [Hash] hgm dados greymass - address, saldo & transacoes
|
143
141
|
# @return [Hash] dados juntos bigquery & greymass
|
@@ -153,14 +151,22 @@ module Cns
|
|
153
151
|
}
|
154
152
|
end
|
155
153
|
|
156
|
-
#
|
154
|
+
# Lazy Greymass API Initialization
|
155
|
+
# @return [Apibc] API instance
|
156
|
+
def api
|
157
|
+
@api ||= Apibc.new
|
158
|
+
end
|
159
|
+
|
160
|
+
# Fetch all Greymass data
|
161
|
+
# @return [Hash] Hash of Greymass data indexed by address
|
157
162
|
def gmd
|
158
|
-
@gmd ||= bqd[:wb].map { |o| bsgm(o) }
|
163
|
+
@gmd ||= bqd[:wb].map { |o| bsgm(o) }.each_with_object({}) { |h, a| a[h[:ax]] = h }
|
159
164
|
end
|
160
165
|
|
161
|
-
#
|
166
|
+
# Fetch combined BigQuery and Greymass data
|
167
|
+
# @return [Array<Hash>] Combined data list
|
162
168
|
def dados
|
163
|
-
@dados ||= bqd[:wb].map { |b| bqgm(b, gmd
|
169
|
+
@dados ||= bqd[:wb].map { |b| bqgm(b, gmd[b[:ax]]) }
|
164
170
|
end
|
165
171
|
|
166
172
|
# @return [Array<Integer>] indices transacoes bigquery
|
@@ -170,12 +176,13 @@ module Cns
|
|
170
176
|
|
171
177
|
# @return [Array<Integer>] indices transacoes novas (greymass - bigquery)
|
172
178
|
def idt
|
173
|
-
@idt ||= gmd.map { |o| o[:tx].map { |i| i[:itx] } }.flatten - bqidt
|
179
|
+
@idt ||= gmd.values.map { |o| o[:tx].map { |i| i[:itx] } }.flatten - bqidt
|
174
180
|
end
|
175
181
|
|
176
|
-
#
|
182
|
+
# Get new transactions
|
183
|
+
# @return [Array<Hash>] List of new transactions
|
177
184
|
def novneost
|
178
|
-
@novneost ||= gmd.map { |t| t[:tx].select { |o| idt.include?(o[:itx]) } }.flatten.uniq { |i| i[:itx] }
|
185
|
+
@novneost ||= gmd.values.map { |t| t[:tx].select { |o| idt.include?(o[:itx]) } }.flatten.uniq { |i| i[:itx] }
|
179
186
|
end
|
180
187
|
end
|
181
188
|
end
|
data/lib/cns/kraken.rb
CHANGED
@@ -16,9 +16,7 @@ module Cns
|
|
16
16
|
# @option pop [Hash] :h ({}) configuracao dias ajuste reposicionamento temporal
|
17
17
|
# @option pop [Boolean] :v (false) mostra dados transacoes trades & ledger?
|
18
18
|
# @option pop [Boolean] :t (false) mostra transacoes todas ou somente novas?
|
19
|
-
# @return [Kraken] API kraken - obter saldos & transacoes trades e ledger
|
20
19
|
def initialize(dad, pop)
|
21
|
-
@api = Apice.new
|
22
20
|
@bqd = dad
|
23
21
|
@ops = pop.transform_keys(&:to_sym)
|
24
22
|
end
|
@@ -136,6 +134,22 @@ module Cns
|
|
136
134
|
hlx.map { |t| t.merge(asset: t[:asset].upcase, amount: t[:amount].to_d, fee: t[:fee].to_d) }
|
137
135
|
end
|
138
136
|
|
137
|
+
# Lazy kraken API Initialization decorated with rate limiting logic
|
138
|
+
# @return [Kraken] API - obter saldos & transacoes trades e ledger
|
139
|
+
def api
|
140
|
+
@api ||=
|
141
|
+
begin
|
142
|
+
t = Apice.new
|
143
|
+
# Rate limiting to this specific instance (0.5s in Kraken)
|
144
|
+
t.define_singleton_method(:run_curl) do |curl, url, **options|
|
145
|
+
sleep(@lapi - Time.now + 0.5) if @lapi && Time.now - @lapi < 0.5
|
146
|
+
super(curl, url, **options)
|
147
|
+
@lapi = Time.now
|
148
|
+
end
|
149
|
+
t
|
150
|
+
end
|
151
|
+
end
|
152
|
+
|
139
153
|
# @return [Hash] dados exchange kraken - saldos & transacoes trades e ledger
|
140
154
|
def usd
|
141
155
|
@usd ||= {sl: pusa(api.account_us), kt: pust(api.trades_us), kl: pusl(api.ledger_us)}
|
data/lib/cns/version.rb
CHANGED
metadata
CHANGED
@@ -1,13 +1,13 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: cns
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.9.
|
4
|
+
version: 0.9.8
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Hernâni Rodrigues Vaz
|
8
8
|
bindir: exe
|
9
9
|
cert_chain: []
|
10
|
-
date: 2025-03-
|
10
|
+
date: 2025-03-07 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: bundler
|
@@ -135,6 +135,20 @@ dependencies:
|
|
135
135
|
- - ">="
|
136
136
|
- !ruby/object:Gem::Version
|
137
137
|
version: '0'
|
138
|
+
- !ruby/object:Gem::Dependency
|
139
|
+
name: faraday-retry
|
140
|
+
requirement: !ruby/object:Gem::Requirement
|
141
|
+
requirements:
|
142
|
+
- - ">="
|
143
|
+
- !ruby/object:Gem::Version
|
144
|
+
version: '0'
|
145
|
+
type: :runtime
|
146
|
+
prerelease: false
|
147
|
+
version_requirements: !ruby/object:Gem::Requirement
|
148
|
+
requirements:
|
149
|
+
- - ">="
|
150
|
+
- !ruby/object:Gem::Version
|
151
|
+
version: '0'
|
138
152
|
- !ruby/object:Gem::Dependency
|
139
153
|
name: google-cloud-bigquery
|
140
154
|
requirement: !ruby/object:Gem::Requirement
|