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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 60e46e5d964570303d66901fde56320d6102a12ff0f53495d420dc73fc5c5045
4
- data.tar.gz: fbc9a980fe5be810ee82c541ecbd769feae8ea87e84a9a2d80c356e0e8b313a0
3
+ metadata.gz: f44706093bb37569dad944e52da3becb0d1bb8ab34e48c9dd30c2f2cd6599f3c
4
+ data.tar.gz: 1cb9e471623d4abbf526a77278e5ff27614e52093d3a79a56229a9c9bf39783f
5
5
  SHA512:
6
- metadata.gz: 425413cbc19a0cbd77889e32a8cac2ba934cf32c9775e2c3bfd891d732f81c025b9f89b92e87211980b4f1e6a318a968fe369b46add18a3c133316b46ec64ceb
7
- data.tar.gz: 3cd4b50b92b93d31abe4795d1887a24dd358ba4d7aee75edb0ff2329a0ff5b722217547dc77b2aaa944b002c3d22b3107b4a561eea4c8209e8a0a6936ad0448f
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.6)
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
- res = es_req('balancemulti', addresses.join(','), 1, tag: 'latest')
26
- res[:status] == '1' ? res[:result] || [] : []
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
- sleep(ofs.zero? ? 0 : 2)
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
- '%<s1>-6.6s %<s2>-42.42s %<v1>13.6f %<v2>13.6f %<ok>-3s',
111
- s1: hjn[:id],
112
- s2: hjn[:ax],
113
- v1: hjn[:es],
114
- v2: hjn[:bs],
115
- ok: ok?(hjn) ? 'OK' : 'NOK'
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
- '%<s1>-6.6s %<s2>-10.10s %<v1>11.4f %<n1>3i %<n2>2i %<n3>2i %<n4>2i %<w1>4i %<v2>11.4f %<n5>3i %<n6>2i %<n7>2i %<n8>2i %<w2>4i %<ok>-3s',
124
- s1: hjn[:id],
125
- s2: foe1(hjn[:ax], 10),
126
- v1: hjn[:es],
127
- n1: hjn[:et].count,
128
- n2: hjn[:ei].count,
129
- n3: hjn[:ep].count,
130
- n4: hjn[:ek].count,
131
- w1: hjn[:ew].count,
132
- v2: hjn[:bs],
133
- n5: hjn[:bt].count,
134
- n6: hjn[:bi].count,
135
- n7: hjn[:bp].count,
136
- n8: hjn[:bk].count,
137
- w2: hjn[:bw].count,
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
- # @param [Hash] htx transacao etherscan normal(t)/(i)nternal
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
- '%<hx>-29.29s %<fr>-15.15s %<to>-15.15s %<dt>10.10s %<vl>7.3f',
195
- hx: foe1(htx[:hash], 29),
196
- fr: foe2(htx[:from], 15),
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
- dt: htx[:timeStamp].strftime('%F'),
199
- vl: htx[:value] / (10**18)
200
+ date: htx[:timeStamp].strftime('%F'),
201
+ value: htx[:value] / (10**18)
200
202
  )
201
203
  end
202
204
 
203
- # @param [Hash] hkx transacao etherscan to(k)en
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
- '%<hx>-20.20s %<fr>-15.15s %<to>-15.15s %<dt>10.10s %<vl>10.3f %<sy>-5.5s',
208
- hx: foe1(hkx[:hash], 20),
209
- fr: foe2(hkx[:from], 15),
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
- dt: hkx[:timeStamp].strftime('%F'),
212
- vl: hkx[:value] / (10**18),
213
- sy: hkx[:tokenSymbol]
214
+ date: hkx[:timeStamp].strftime('%F'),
215
+ value: hkx[:value] / (10**18),
216
+ symbol: hkx[:tokenSymbol]
214
217
  )
215
218
  end
216
219
 
217
- # @param [Hash] hpx transacao etherscan (p)roduced blocks
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('%<bn>9i %<fr>-41.41s %<dt>10.10s %<vl>17.6f', bn: hpx[:blockNumber], fr: foe2(hpx[:iax], 41), dt: hpx[:timeStamp].strftime('%F'), vl: hpx[:blockReward] / (10**18))
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
- # @param [Hash] hwx transacao etherscan (w)ithdrawals
224
- # @return [String] texto formatado transacao withdrawals etherscan
233
+ # Format (w)ithdrawal transaction
234
+ # @param [Hash] hwx transacao etherscan
235
+ # @return [String] texto formatado
225
236
  def fow(hwx)
226
- format('%<bn>10i %<vi>9i %<dt>10.10s %<vl>10.6f', bn: hwx[:withdrawalIndex], vi: hwx[:validatorIndex], dt: hwx[:timeStamp].strftime('%F'), vl: hwx[:amount] / (10**9))
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 normal(t)/(i)nternal/to(k)en
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 (p)roduced blocks
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 (w)ithdrawals
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
- # @return [Array<Hash>] todos os dados etherscan - saldos & transacoes
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.find { |e| b[:ax] == e[:ax] }) }
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
- # @return [Array<Hash>] lista transacoes normais novas
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
- # @return [Array<Hash>] lista transacoes internas novas
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
- # @return [Array<Hash>] lista transacoes block novas
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
- # @return [Array<Hash>] lista transacoes withdrawals novas
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
- # @return [Array<Hash>] lista transacoes token novas
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
- # @return [String] texto carteiras & transacoes & ajuste dias
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 { |o| puts(foct(o)) }
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
- ntx = send(TT[:new])
49
- return unless ops[:v] && ntx.any?
40
+ return unless ops[:v] && novneost.any?
50
41
 
51
- puts(TT[:header])
52
- ntx.sort_by { |s| -s[TT[:sork]] }.each { |t| puts(send(TT[:format], t)) }
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
- ntx = send(TT[:new])
58
- return unless ntx.any?
48
+ return unless novneost.any?
59
49
 
60
- puts("\nstring ajuste dias\n-h=#{ntx.sort_by { |s| -s[TT[:sork]] }.map { |t| "#{t[TT[:adjk]]}:0" }.join(' ')}")
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
- '%<s1>-12.12s %<v1>14.4f %<n1>4i %<v2>14.4f %<n2>4i %<ok>-3s',
68
- s1: hjn[:ax],
69
- v1: hjn[:es],
70
- n1: hjn[:et].count,
71
- v2: hjn[:bs],
72
- n2: hjn[:bt].count,
73
- ok: ok?(hjn) ? 'OK' : 'NOK'
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
- '%<bn>12i %<fr>-12.12s %<to>-12.12s %<ac>-10.10s %<dt>10.10s %<vl>12.4f %<sy>-6.6s',
88
- ac: hlx[:name],
89
- fr: hlx[:from],
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
- dt: hlx[:block_time].strftime('%F'),
94
- sy: hlx[:moe]
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
- # @return [Array<Hash>] todos os dados greymass - saldos & transacoes
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
- # @return [Array<Hash>] todos os dados juntos bigquery & greymass
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.find { |g| b[:ax] == g[:ax] }) }
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
- # @return [Array<Hash>] lista transacoes novas
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
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Cns
4
- VERSION = '0.9.6'
4
+ VERSION = '0.9.8'
5
5
  end
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.6
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-06 00:00:00.000000000 Z
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