cns 0.9.3 → 0.9.5

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: f5864db949cd54475943852878f488bcb0e62e342dafcc2f0be4d2e418a6e4a8
4
- data.tar.gz: 2f33d2a740c1787f92246625ff11d44f1cc06d4ac8cbe2a54e610aca6a2309ed
3
+ metadata.gz: 5e6f05fc2c2fc0c7c1e34cda63d18338f37db37957be5f1af27c60041cd761b5
4
+ data.tar.gz: bab915764960144e4b713348958b173524f81ba42eb53d2d0df1242c2721d194
5
5
  SHA512:
6
- metadata.gz: 989d84cf9d660f93068f135d35f8963633431787446a2a666562230560965711430820cc57ebc8ec7e6dd540005c920d1791431ac20e9acd0a71e84565a24dc4
7
- data.tar.gz: b8818edca07d3e902cd1a126e9474efe1ff15c3088ca6939f27be8f5422fd91f88a024e2b407a75dd156426cbb4f30b9971548c23ccad0f648348607bd6da435
6
+ metadata.gz: 78336dc2f0cc71f52a0fffcdc0a520e62fb8b77bee5173a18c21de958f0f17df2a7042d21a344a324b1cc3861bab978b6e3a66b96aba66981bd147571e8a5fdc
7
+ data.tar.gz: cd747260647643fc61a7ac298a716489df77f71a130571f3360efe5a4268a7e27644150b6c7649d395fd900777439e78d98892a315c5423ba4fd8649eb3c2fd1
data/Gemfile.lock CHANGED
@@ -1,7 +1,7 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- cns (0.9.3)
4
+ cns (0.9.5)
5
5
  curb
6
6
  faraday
7
7
  google-cloud-bigquery
@@ -79,7 +79,7 @@ GEM
79
79
  google-cloud-errors (~> 1.0)
80
80
  google-cloud-env (2.2.1)
81
81
  faraday (>= 1.0, < 3.a)
82
- google-cloud-errors (1.4.0)
82
+ google-cloud-errors (1.5.0)
83
83
  google-logging-utils (0.1.0)
84
84
  googleauth (1.13.1)
85
85
  faraday (>= 1.0, < 3.a)
@@ -137,7 +137,7 @@ GEM
137
137
  reverse_markdown (3.0.0)
138
138
  nokogiri
139
139
  rexml (3.4.1)
140
- rubocop (1.73.1)
140
+ rubocop (1.73.2)
141
141
  json (~> 2.3)
142
142
  language_server-protocol (~> 3.17.0.2)
143
143
  lint_roller (~> 1.1.0)
data/lib/cns/apibc.rb CHANGED
@@ -7,141 +7,156 @@ require('json')
7
7
  module Cns
8
8
  # classe para acesso dados blockchains
9
9
  class Apibc
10
- ESPS = 10_000
11
- GMPS = 100
10
+ ESPS = 10_000 # Etherscan maximum records per page
11
+ GMPS = 100 # Greymass maximum actions per request
12
12
 
13
13
  def initialize
14
- @etherscan_conn = connection('https://api.etherscan.io')
15
- @greymass_conn = connection('https://eos.greymass.com')
14
+ @escn = connection('https://api.etherscan.io')
15
+ @gmcn = connection('https://eos.greymass.com')
16
+ @esky = ENV.fetch('ETHERSCAN_API_KEY', nil)
16
17
  end
17
18
 
18
19
  # Get account balances for multiple ETH addresses
19
- # @param addresses [Array<String>] List of ETH addresses (max 20)
20
+ # @param [Array<String>] addresses List of ETH addresses (max 20)
20
21
  # @return [Array<Hash>] List of addresses with balances
21
22
  def account_es(addresses)
22
- response = etherscan_req('balancemulti', addresses.join(','), 1, tag: 'latest')
23
- response[:status] == '1' ? response[:result] || [] : []
23
+ res = es_req('balancemulti', addresses.join(','), 1, tag: 'latest')
24
+ res[:status] == '1' ? res[:result] || [] : []
24
25
  end
25
26
 
26
27
  # Get normal transactions for ETH address
27
28
  # @param [String] address endereco ETH
28
29
  # @return [Array<Hash>] lista transacoes normais etherscan
29
30
  def norml_es(address)
30
- pag_etherscan_req('txlist', address)
31
+ pag_es_req('txlist', address)
31
32
  end
32
33
 
33
34
  # Get internal transactions for ETH address
34
35
  # @param (see norml_es)
35
36
  # @return [Array<Hash>] lista transacoes internas etherscan
36
37
  def inter_es(address)
37
- pag_etherscan_req('txlistinternal', address)
38
+ pag_es_req('txlistinternal', address)
38
39
  end
39
40
 
40
41
  # Get mined blocks for ETH address
41
42
  # @param (see norml_es)
42
43
  # @return [Array<Hash>] lista blocos etherscan
43
44
  def block_es(address)
44
- pag_etherscan_req('getminedblocks', address, blocktype: 'blocks')
45
+ pag_es_req('getminedblocks', address, blocktype: 'blocks')
45
46
  end
46
47
 
47
48
  # Get withdrawals for ETH address
48
49
  # @param (see norml_es)
49
50
  # @return [Array<Hash>] lista blocos etherscan
50
51
  def withw_es(address)
51
- pag_etherscan_req('txsBeaconWithdrawal', address)
52
+ pag_es_req('txsBeaconWithdrawal', address)
52
53
  end
53
54
 
54
55
  # Get token transfers for ETH address
55
56
  # @param (see norml_es)
56
57
  # @return [Array<Hash>] lista token transfer events etherscan
57
58
  def token_es(address)
58
- pag_etherscan_req('tokentx', address)
59
+ pag_es_req('tokentx', address)
59
60
  end
60
61
 
61
62
  # Get EOS account information
62
- # @param address [String] EOS account name
63
+ # @param [String] address EOS account name
63
64
  # @return [Hash] Account details with resources
64
65
  def account_gm(address)
65
- response = greymass_req('/v1/chain/get_account', account_name: address)
66
- response[:core_liquid_balance]&.to_d&.positive? ? response : gm_erro
66
+ res = gm_req('/v1/chain/get_account', account_name: address)
67
+ res[:core_liquid_balance]&.to_d&.positive? ? res : gm_erro
67
68
  end
68
69
 
69
70
  # Get complete transaction history for EOS account
70
71
  # @param (see account_gm)
71
72
  # @return [Array<Hash>] lista completa transacoes greymass
72
73
  def ledger_gm(address)
73
- actions = []
74
+ trx = []
74
75
  pos = 0
75
76
  loop do
76
- response = greymass_req('/v1/history/get_actions', account_name: address, pos: pos, offset: GMPS)
77
- batch = response[:actions] || []
78
- actions.concat(batch)
79
- break if batch.size < GMPS
77
+ res = gm_req('/v1/history/get_actions', account_name: address, pos: pos, offset: GMPS)
78
+ bth = res[:actions] || []
79
+ trx.concat(bth)
80
+ break if bth.size < GMPS
80
81
 
81
82
  pos += GMPS
82
83
  end
83
- actions
84
+ trx
84
85
  end
85
86
 
86
87
  private
87
88
 
88
- # Reusable Faraday connection
89
- def connection(base_url)
90
- Faraday.new(base_url) do |conn|
91
- conn.request(:json) # Automatically encodes body as JSON
92
- conn.headers = {content_type: 'application/json', accept: 'application/json', user_agent: 'blockchain-api-client'}
93
- conn.options.timeout = 30
94
- conn.options.open_timeout = 10
95
- conn.adapter(Faraday.default_adapter)
96
- end
97
- end
98
-
99
- # Generic Etherscan API request handler
100
- def etherscan_req(action, address, page = 1, params = {})
101
- params = {module: 'account', action: action, address: address, page: page, apikey: ENV.fetch('ETHERSCAN_API_KEY')}.merge(params)
102
- parse_json(@etherscan_conn.get('/api', params))
89
+ # Make a request to the Etherscan API
90
+ # @param [String] act API action name
91
+ # @param [String] add Blockchain address
92
+ # @param [Integer] pag Page number for pagination
93
+ # @param [Hash] prm Additional request parameters
94
+ # @return [Hash] Parsed API response
95
+ def es_req(act, add, pag = 1, prm = {})
96
+ prm = {module: 'account', action: act, address: add, page: pag, apikey: @esky}.merge(prm)
97
+ parse_json(@escn.get('/api', prm))
103
98
  rescue Faraday::Error
104
99
  {status: '0'}
105
100
  end
106
101
 
107
- # Generic method for paginated Etherscan requests
108
- # @param action [String] API action name
109
- # @param address [String] Blockchain address
110
- # @param params [Hash] Additional request parameters
102
+ # Fetch paginated results from Etherscan
103
+ # @param act [String] API action name
104
+ # @param add [String] Blockchain address
105
+ # @param prm [Hash] Additional request parameters
111
106
  # @return [Array<Hash>] Combined results from all pages
112
- def pag_etherscan_req(action, address, params = {})
113
- results = []
114
- page = 1
107
+ def pag_es_req(act, add, prm = {})
108
+ prm = prm.merge(offset: ESPS)
109
+ trx = []
110
+ pag = 1
115
111
  loop do
116
- response = etherscan_req(action, address, page, params)
117
- break unless response[:status] == '1'
112
+ res = es_req(act, add, pag, prm)
113
+ break unless res[:status] == '1'
118
114
 
119
- batch = response[:result] || []
120
- results.concat(batch)
121
- break if batch.size < ESPS
115
+ bth = res[:result] || []
116
+ trx.concat(bth)
117
+ break if bth.size < ESPS
122
118
 
123
- page += 1
119
+ pag += 1
124
120
  end
125
- results
126
- end
127
-
128
- # Generic Greymass API error
129
- def gm_erro
130
- {core_liquid_balance: 0, total_resources: {net_weight: 0, cpu_weight: 0}}
121
+ trx
131
122
  end
132
123
 
133
- # Generic Greymass API request handler
134
- def greymass_req(endpoint, payload)
135
- parse_json(@greymass_conn.post(endpoint) { |req| req.body = payload })
124
+ # Make a request to the Greymass API
125
+ # @param [String] url API endpoint
126
+ # @param [Hash] pyl Request payload
127
+ # @return [Hash] Parsed API response
128
+ def gm_req(url, pyl)
129
+ parse_json(@gmcn.post(url) { |r| r.body = pyl })
136
130
  rescue Faraday::Error
137
131
  gm_erro
138
132
  end
139
133
 
140
- # Safe JSON parsing with error handling
134
+ # Default error response for Greymass API
135
+ # @return [Hash] Error response with zeroed values
136
+ def gm_erro
137
+ {core_liquid_balance: 0, total_resources: {net_weight: 0, cpu_weight: 0}}
138
+ end
139
+
140
+ # Safely parse JSON response
141
+ # @param [Faraday::Response] res API response
142
+ # @return [Hash] Parsed JSON or empty hash on error
141
143
  def parse_json(res)
142
144
  JSON.parse(res.body, symbolize_names: true) || {}
143
145
  rescue JSON::ParserError
144
146
  {}
145
147
  end
148
+
149
+ # Create a Faraday connection with JSON configuration
150
+ # @param [String] url Base URL for the API
151
+ # @return [Faraday::Connection] Configured Faraday connection
152
+ def connection(url)
153
+ Faraday.new(url) do |conn|
154
+ conn.request(:json)
155
+ conn.headers = {accept: 'application/json', user_agent: 'blockchain-api-client'}
156
+ conn.options.timeout = 30
157
+ conn.options.open_timeout = 10
158
+ conn.adapter(Faraday.default_adapter)
159
+ end
160
+ end
146
161
  end
147
162
  end
data/lib/cns/apice.rb CHANGED
@@ -17,8 +17,13 @@ module Cns
17
17
  @curl.connect_timeout = 10
18
18
  @curl.follow_location = true
19
19
  @curl.ssl_verify_peer = true
20
+ @deky = ENV.fetch('BITCOINDE_API_KEY', nil)
21
+ @desc = ENV.fetch('BITCOINDE_API_SECRET', nil)
22
+ @usky = ENV.fetch('KRAKEN_API_KEY', nil)
23
+ @ussc = ENV.fetch('KRAKEN_API_SECRET', nil)
20
24
  end
21
25
 
26
+ # Get account balances from Bitcoin.de
22
27
  # @return [Hash] saldos no bitcoinde
23
28
  def account_de
24
29
  uri = "#{API[:de]}/account"
@@ -28,6 +33,7 @@ module Cns
28
33
  {}
29
34
  end
30
35
 
36
+ # Get trades from Bitcoin.de
31
37
  # @return [Array<Hash>] trades bitcoinde
32
38
  def trades_de
33
39
  pag_de_req("#{API[:de]}/trades", {state: 1}, :trades)
@@ -35,6 +41,7 @@ module Cns
35
41
  []
36
42
  end
37
43
 
44
+ # Get deposits from Bitcoin.de, uniformly formatted
38
45
  # @return [Array<Hash>] depositos uniformizados bitcoinde
39
46
  def deposits_de
40
47
  pag_de_req("#{API[:de]}/btc/deposits", {state: 2}, :deposits) { |i| i.map { |h| deposit_unif(h) } }
@@ -42,6 +49,7 @@ module Cns
42
49
  []
43
50
  end
44
51
 
52
+ # Get withdrawals from Bitcoin.de, uniformly formatted
45
53
  # @return [Array<Hash>] withdrawals uniformizadas bitcoinde
46
54
  def withdrawals_de
47
55
  pag_de_req("#{API[:de]}/btc/withdrawals", {state: 1}, :withdrawals) { |i| i.map { |h| withdrawal_unif(h) } }
@@ -49,60 +57,83 @@ module Cns
49
57
  []
50
58
  end
51
59
 
60
+ # Get account balances from Kraken
52
61
  # @return [Hash] saldos kraken
53
62
  def account_us
54
63
  uri = 'Balance'
55
64
  ops = {nonce: nnc}
56
- run_curl(@curl, "#{API[:us]}/#{uri}", method: :post, post_data: ops, headers: hus(uri, ops))
65
+ run_curl(@curl, "#{API[:us]}/#{uri}", method: 'POST', post_data: ops, headers: hus(uri, ops))
57
66
  parse_json(@curl).fetch(:result, {})
58
67
  rescue Curl::Err::CurlError
59
68
  {}
60
69
  end
61
70
 
71
+ # Get trades from Kraken
62
72
  # @return [Hash] trades kraken
63
73
  def trades_us
64
74
  pag_us_req('TradesHistory', :trades)
65
75
  rescue Curl::Err::CurlError
66
- {}
76
+ []
67
77
  end
68
78
 
79
+ # Get ledger from Kraken
69
80
  # @return [Hash] ledger kraken
70
81
  def ledger_us
71
82
  pag_us_req('Ledgers', :ledger)
72
83
  rescue Curl::Err::CurlError
73
- {}
84
+ []
74
85
  end
75
86
 
76
87
  private
77
88
 
89
+ # Uniformly format transacao kraken
90
+ # @param [Symbol] key id da transacao
91
+ # @param [Hash] trx transacao
92
+ # @return [Hash] transacao uniformizada
93
+ def us_unif(key, trx)
94
+ t = Integer(trx[:time])
95
+ trx.merge(txid: key.to_s, srx: t, time: Time.at(t))
96
+ rescue StandardError
97
+ {}
98
+ end
99
+
78
100
  # Generic paginated request handler for Kraken
101
+ # @param [String] uri API endpoint URI
102
+ # @param [Symbol] key Key to extract from the result
103
+ # @yield [Array<Hash>] Block to process each batch of results
104
+ # @return [Array<Hash>] Combined results from all pages
79
105
  def pag_us_req(uri, key)
80
- has = {}
106
+ ary = []
81
107
  ofs = 0
82
108
  loop do
83
109
  sleep(ofs.zero? ? 0 : 2)
84
110
  ops = {nonce: nnc, ofs: ofs}
85
- run_curl(@curl, "#{API[:us]}/#{uri}", method: :post, post_data: ops, headers: hus(uri, ops))
86
- batch = parse_json(@curl).fetch(:result, {}).fetch(key, [])
87
- break if batch.empty?
111
+ run_curl(@curl, "#{API[:us]}/#{uri}", method: 'POST', post_data: ops, headers: hus(uri, ops))
112
+ bth = parse_json(@curl).fetch(:result, {}).fetch(key, []).map { |k, v| us_unif(k, v) }
113
+ break if bth.empty?
88
114
 
89
- has.merge!(batch)
90
- ofs += batch.size
115
+ ary.concat(bth)
116
+ ofs += bth.size
91
117
  end
92
- has
118
+ ary
93
119
  end
94
120
 
95
121
  # Generic paginated request handler for Bitcoin.de
96
- def pag_de_req(base_url, params, key)
122
+ # @param [String] uri Base URL for the API request
123
+ # @param [Hash] prm Additional parameters for the request
124
+ # @param [Symbol] key Key to extract from the result
125
+ # @yield [Array<Hash>] Optional block to process each batch of results
126
+ # @return [Array<Hash>] Combined results from all pages
127
+ def pag_de_req(uri, prm, key)
97
128
  ary = []
98
129
  pag = 1
99
130
  loop do
100
- url = "#{base_url}?#{URI.encode_www_form(params.merge(page: pag))}"
131
+ url = "#{uri}?#{URI.encode_www_form(prm.merge(page: pag))}"
101
132
  run_curl(@curl, url, headers: hde(url))
102
- result = parse_json(@curl)
103
- batch = result.fetch(key, [])
104
- ary.concat(block_given? ? yield(batch) : batch)
105
- break if result[:page]&.[](:current)&.>= result[:page]&.[](:last)
133
+ res = parse_json(@curl)
134
+ bth = res.fetch(key, [])
135
+ ary.concat(block_given? ? yield(bth) : bth)
136
+ break if res[:page]&.[](:current)&.>= res[:page]&.[](:last)
106
137
 
107
138
  pag += 1
108
139
  end
@@ -110,64 +141,76 @@ module Cns
110
141
  end
111
142
 
112
143
  # Configure Curl object for request
113
- def run_curl(curl, url, method: :get, post_data: nil, headers: {})
144
+ # @param [Curl::Easy] curl Curl object to configure
145
+ # @param [String] url URL for the request
146
+ # @param [String] method HTTP method (GET or POST)
147
+ # @param [Hash] post_data Data to send in POST requests
148
+ # @param [Hash] headers HTTP headers for the request
149
+ def run_curl(curl, url, method: 'GET', post_data: nil, headers: {})
114
150
  curl.reset
115
151
  curl.url = url
116
- curl.http(method == :post ? 'POST' : 'GET')
152
+ curl.http(method)
117
153
  curl.headers = headers
118
154
  curl.post_body = URI.encode_www_form(post_data) if post_data
119
155
  curl.perform
120
156
  end
121
157
 
122
158
  # Safe JSON parsing with error handling
159
+ # @param [Curl::Easy] res Curl response object
160
+ # @return [Hash] Parsed JSON or empty hash on error
123
161
  def parse_json(res)
124
162
  JSON.parse(res.body_str, symbolize_names: true)
125
163
  rescue JSON::ParserError
126
164
  {}
127
165
  end
128
166
 
129
- # @return [Integer] continually-increasing unsigned integer nonce from the current Unix Time
167
+ # Generate a continually-increasing unsigned integer nonce from the current Unix Time
168
+ # @return [Integer] Nonce value
130
169
  def nnc
131
170
  Integer(Float(Time.now) * 1e6)
132
171
  end
133
172
 
173
+ # Uniformly format a deposit from Bitcoin.de
174
+ # @param [Hash] has Deposit data from Bitcoin.de
134
175
  # @return [Hash] deposito uniformizado bitcoinde
135
176
  def deposit_unif(has)
136
- {add: has[:address], time: Time.parse(has[:created_at]), qt: has[:amount], txid: Integer(has[:deposit_id])}.merge(tp: 'deposit', moe: 'btc', fee: '0')
177
+ {add: has[:address], time: Time.parse(has[:created_at]), qtd: has[:amount].to_d, nxid: Integer(has[:deposit_id])}.merge(tp: 'deposit', moe: 'BTC', fee: 0.to_d)
137
178
  end
138
179
 
180
+ # Uniformly format a withdrawal from Bitcoin.de
181
+ # @param [Hash] has Withdrawal data from Bitcoin.de
139
182
  # @return [Hash] withdrawal uniformizada bitcoinde
140
183
  def withdrawal_unif(has)
141
184
  {
142
185
  add: has[:address],
143
186
  time: Time.parse(has[:transferred_at]),
144
- qt: has[:amount],
145
- fee: has[:network_fee],
146
- txid: Integer(has[:withdrawal_id]),
187
+ qtd: -1 * has[:amount].to_d,
188
+ fee: has[:network_fee].to_d,
189
+ nxid: Integer(has[:withdrawal_id]),
147
190
  tp: 'withdrawal',
148
- moe: 'btc'
191
+ moe: 'BTC'
149
192
  }
150
193
  end
151
194
 
152
- # @param [String] qde query a incluir no pedido HTTP
153
- # @param [Integer] non continually-increasing unsigned integer
154
- # @return [Hash] headers necessarios para pedido HTTP da exchange bitcoinde
195
+ # Generate headers for Bitcoin.de HTTP requests
196
+ # @param [String] qde Query string to include in the HTTP request
197
+ # @param [Integer] non Nonce value (default: generated from nnc)
198
+ # @return [Hash] Headers required for Bitcoin.de HTTP requests
155
199
  def hde(qde, non = nnc)
156
- key = ENV.fetch('BITCOINDE_API_KEY', nil)
157
- md5 = ['GET', qde, key, non, Digest::MD5.hexdigest('')].join('#')
158
- mac = OpenSSL::HMAC.hexdigest('sha256', ENV.fetch('BITCOINDE_API_SECRET', nil), md5)
159
- {'X-API-KEY': key, 'X-API-NONCE': non, 'X-API-SIGNATURE': mac}
200
+ md5 = ['GET', qde, @deky, non, Digest::MD5.hexdigest('')].join('#')
201
+ mac = OpenSSL::HMAC.hexdigest('sha256', @desc, md5)
202
+ {'X-API-KEY' => @deky, 'X-API-NONCE' => non.to_s, 'X-API-SIGNATURE' => mac}
160
203
  end
161
204
 
205
+ # Generate headers for Kraken HTTP requests
162
206
  # @param [String] qus query a incluir no pedido HTTP
163
207
  # @param [Hash] ops opcoes trabalho
164
208
  # @option ops [Hash] :nonce continually-increasing unsigned integer
165
- # @return [Hash] headers necessarios para pedido HTTP da exchange kraken
209
+ # @return [Hash] Headers required for Kraken HTTP requests
166
210
  def hus(qus, ops)
167
- key = ENV.fetch('KRAKEN_API_KEY', nil)
168
211
  sha = ['/0/private/', qus, Digest::SHA256.digest("#{ops[:nonce]}#{URI.encode_www_form(ops)}")].join
169
- mac = OpenSSL::HMAC.digest('sha512', Base64.decode64(ENV.fetch('KRAKEN_API_SECRET', nil)), sha)
170
- {'api-key': key, 'api-sign': Base64.strict_encode64(mac)}
212
+ mac = OpenSSL::HMAC.digest('sha512', Base64.decode64(@ussc), sha)
213
+ {'api-key' => @usky, 'api-sign' => Base64.strict_encode64(mac)}
171
214
  end
172
215
  end
173
216
  end
data/lib/cns/bigquery.rb CHANGED
@@ -9,16 +9,16 @@ module Cns
9
9
  FO = File.expand_path("~/#{File.basename($PROGRAM_NAME)}.log")
10
10
  EM = %i[EOS XETH ZEUR btc eth]
11
11
  TB = {
12
- netht: %w[blocknumber timestamp txhash nonce blockhash transactionindex axfrom axto iax value gas gasprice gasused iserror txreceipt_status input contractaddress dias],
13
- hetht: %i[blockNumber timeStamp hash nonce blockHash transactionIndex from to iax value gas gasPrice gasUsed isError txreceipt_status input contractAddress],
14
- nethi: %w[blocknumber timestamp txhash axfrom axto iax value contractaddress input type gas gasused traceid iserror errcode dias],
15
- hethi: %i[blockNumber timeStamp hash from to iax value contractAddress input type gas gasUsed traceId isError errCode],
12
+ netht: %w[txhash blocknumber timestamp nonce blockhash transactionindex axfrom axto iax value gas gasprice gasused iserror txreceipt_status input contractaddress dias],
13
+ hetht: %i[hash blockNumber timeStamp nonce blockHash transactionIndex from to iax value gas gasPrice gasUsed isError txreceipt_status input contractAddress],
14
+ nethi: %w[txhash blocknumber timestamp axfrom axto iax value contractaddress input type gas gasused traceid iserror errcode dias],
15
+ hethi: %i[hash blockNumber timeStamp from to iax value contractAddress input type gas gasUsed traceId isError errCode],
16
16
  nethp: %w[blocknumber timestamp blockreward iax dias],
17
17
  hethp: %i[blockNumber timeStamp blockReward iax],
18
18
  nethw: %w[withdrawalindex validatorindex address amount blocknumber timestamp dias],
19
19
  hethw: %i[withdrawalIndex validatorIndex address amount blockNumber timeStamp],
20
- nethk: %w[blocknumber timestamp txhash nonce blockhash transactionindex axfrom axto iax value tokenname tokensymbol tokendecimal gas gasprice gasused input contractaddress dias],
21
- hethk: %i[blockNumber timeStamp hash nonce blockHash transactionIndex from to iax value tokenName tokenSymbol tokenDecimal gas gasPrice gasUsed input contractAddress],
20
+ nethk: %w[txhash blocknumber timestamp nonce blockhash transactionindex axfrom axto iax value tokenname tokensymbol tokendecimal gas gasprice gasused input contractaddress dias],
21
+ hethk: %i[hash blockNumber timeStamp nonce blockHash transactionIndex from to iax value tokenName tokenSymbol tokenDecimal gas gasPrice gasUsed input contractAddress],
22
22
  neost: %w[gseq aseq bnum time contract action acfrom acto iax amount moeda memo dias],
23
23
  heost: %i[global_action_seq account_action_seq block_num block_time account name from to iax quantity moe memo],
24
24
  cdet: %w[txid time tp user btc eur dtc dias],
@@ -30,6 +30,22 @@ module Cns
30
30
  cusl: %w[txid refid time type aclass asset amount fee],
31
31
  husl: %i[txid refid time type aclass asset amount fee]
32
32
  }
33
+ # para testes bigquery
34
+ TL = {
35
+ ins: 'INSERT',
36
+ exo: false,
37
+ est: '', # ' limit 226',
38
+ esi: '', # ' limit 22',
39
+ esp: '', # ' limit 72',
40
+ esw: '', # ' limit 2299',
41
+ esk: '', # ' limit 20',
42
+ gmt: '', # ' limit 1091',
43
+ ust: '', # ' limit 182',
44
+ usl: '', # ' limit 448',
45
+ det: '', # ' limit 27',
46
+ del: '' # ' limit 16'
47
+
48
+ }
33
49
 
34
50
  # classe para processar bigquery
35
51
  class Bigquery
@@ -154,11 +170,11 @@ module Cns
154
170
  Etherscan.new(
155
171
  {
156
172
  wb: sql("SELECT * FROM #{BD}.wet#{prx[-1]} ORDER BY ax"),
157
- ni: sql("SELECT * FROM #{BD}.#{prx}i"),
158
- nk: sql("SELECT * FROM #{BD}.#{prx}k"),
159
- np: sql("SELECT * FROM #{BD}.#{prx}p"),
160
- nt: sql("SELECT * FROM #{BD}.#{prx}t"),
161
- nw: sql("SELECT * FROM #{BD}.#{prx}w")
173
+ nt: sql("SELECT * FROM #{BD}.#{prx}t#{TL[:est]}"),
174
+ ni: sql("SELECT * FROM #{BD}.#{prx}i#{TL[:esi]}"),
175
+ np: sql("SELECT * FROM #{BD}.#{prx}p#{TL[:esp]}"),
176
+ nw: sql("SELECT * FROM #{BD}.#{prx}w#{TL[:esw]}"),
177
+ nk: sql("SELECT * FROM #{BD}.#{prx}k#{TL[:esk]}")
162
178
  },
163
179
  ops
164
180
  )
@@ -176,32 +192,39 @@ module Cns
176
192
 
177
193
  # @return [Greymass] API blockchain EOS
178
194
  def apigm
179
- @apigm ||= Greymass.new({wb: sql("select * from #{BD}.weos ORDER BY ax"), nt: sql("select * from #{BD}.neosx")}, ops)
195
+ @apigm ||= Greymass.new({wb: sql("select * from #{BD}.weos ORDER BY ax"), nt: sql("select * from #{BD}.neosx#{TL[:gmt]}")}, ops)
180
196
  end
181
197
 
182
198
  # @return [Kraken] API exchange kraken
183
199
  def apius
184
- @apius ||= Kraken.new({sl: sql("select * from #{BD}.cuss").first, nt: sql("select * from #{BD}.cust"), nl: sql("select * from #{BD}.cusl")}, ops)
200
+ @apius ||= Kraken.new({sl: sql("select * from #{BD}.cuss").first, nt: sql("select * from #{BD}.cust#{TL[:ust]}"), nl: sql("select * from #{BD}.cusl#{TL[:usl]}")}, ops)
185
201
  end
186
202
 
187
203
  # @return [Bitcoinde] API exchange bitcoinde
188
204
  def apide
189
- @apide ||= Bitcoinde.new({sl: sql("select * from #{BD}.cdes").first, nt: sql("select * from #{BD}.cdet"), nl: sql("select * from #{BD}.cdel")}, ops)
205
+ @apide ||= Bitcoinde.new({sl: sql("select * from #{BD}.cdes").first, nt: sql("select * from #{BD}.cdet#{TL[:det]}"), nl: sql("select * from #{BD}.cdel#{TL[:del]}")}, ops)
190
206
  end
191
207
 
192
208
  # @return [String] comando insert SQL formatado
193
209
  def ins_sql(tbl, lin)
194
- "INSERT #{BD}.#{tbl} (#{TB[tbl].join(',')}) VALUES #{lin.map { |i| send("#{tbl}_val", i) }.join(',')}"
210
+ # para testes bigquery
211
+ if TL[:exo]
212
+ exl = lin.map { |i| send("#{tbl}_val", i)[1..-2] }
213
+ exi = exl.map { |f| f.split(',').first }.join(',')
214
+ exo = "SELECT #{TB[tbl].join(',')} FROM #{BD}.#{tbl} WHERE #{TB[tbl].first} IN (#{exi}) union all select "
215
+ puts(exo + exl.join(' union all select ') + ' order by 1')
216
+ end
217
+ "#{TL[:ins]} #{BD}.#{tbl} (#{TB[tbl].join(',')}) VALUES #{lin.map { |i| send("#{tbl}_val", i) }.join(',')}"
195
218
  end
196
219
 
197
220
  # @return [String] relatorio execucao dml
198
221
  def dml_out(src, str, ltb)
199
222
  str.concat(
200
- ltb.filter_map do |itm|
201
- novx = src.send("nov#{itm}")
223
+ ltb.filter_map do |i|
224
+ novx = src.send("nov#{i}")
202
225
  next if novx.empty?
203
226
 
204
- format(' %<n>i %<t>s', n: dml(ins_sql(itm, %i[cust cusl].include?(itm) ? novx.values : novx)), t: "#{itm}")
227
+ format(' %<n>i %<t>s', n: dml(ins_sql(i, novx)), t: "#{i}")
205
228
  end
206
229
  )
207
230
  str.join
@@ -290,7 +313,7 @@ module Cns
290
313
  # @param [Hash] htx ledger greymass
291
314
  # @return [String] valores formatados neost
292
315
  def neost_val(htx)
293
- fvals(htx, TB[:heost], :itx)
316
+ fvals(htx, TB[:heost], :global_action_seq)
294
317
  end
295
318
 
296
319
  # @param [Hash] htx trades bitcoinde
@@ -308,7 +331,7 @@ module Cns
308
331
  # @param [Hash] htx trades kraken
309
332
  # @return [String] valores formatados cust
310
333
  def cust_val(htx)
311
- fvals(htx.merge(ledgers: apius.exd[:kl].select { |_, o| o[:refid] == htx[:txid] }.keys.join(',')), TB[:hust], :txid)
334
+ fvals(htx.merge(ledgers: apius.uskl.select { |o| o[:refid] == htx[:txid] }.map { |t| t[:txid] }.join(',')), TB[:hust], :txid)
312
335
  end
313
336
 
314
337
  # @param [Hash] htx ledger kraken
data/lib/cns/bitcoinde.rb CHANGED
@@ -23,15 +23,10 @@ module Cns
23
23
  @ops = pop.transform_keys(&:to_sym)
24
24
  end
25
25
 
26
- # @return [Hash] dados exchange bitcoinde - saldos & trades & deposits & withdrawals
27
- def exd
28
- @exd ||= {sl: pdea(api.account_de), tt: pdet(api.trades_de), tl: pdel(api.deposits_de + api.withdrawals_de)}
29
- end
30
-
31
26
  # @return [String] texto saldos & transacoes & ajuste dias
32
27
  def mresumo
33
28
  puts("\nBITCOINDE\ntipo bitcoinde bigquery")
34
- exd[:sl].sort.each { |k, v| puts(fos(k, v)) }
29
+ ded[:sl].sort.each { |k, v| puts(fos(k, v)) }
35
30
  mtotais
36
31
 
37
32
  mtrades
@@ -45,9 +40,9 @@ module Cns
45
40
 
46
41
  # mosta contadores transacoes
47
42
  def mtotais
48
- vtt = exd[:tt].count
43
+ vtt = ded[:tt].count
49
44
  vnt = bqd[:nt].count
50
- vtl = exd[:tl].count
45
+ vtl = ded[:tl].count
51
46
  vnl = bqd[:nl].count
52
47
 
53
48
  puts("TRADES #{format('%<a>20i %<b>21i %<o>3.3s', a: vtt, b: vnt, o: vtt == vnt ? 'OK' : 'NOK')}")
@@ -105,7 +100,7 @@ module Cns
105
100
  def fol(hlx)
106
101
  format(
107
102
  '%<ky>6i %<dt>19.19s %<ty>-10.10s %<mo>-3.3s %<pr>19.8f %<vl>18.8f',
108
- ky: hlx[:txid],
103
+ ky: hlx[:nxid],
109
104
  dt: hlx[:time].strftime('%F %T'),
110
105
  ty: hlx[:tp],
111
106
  mo: hlx[:moe],
@@ -114,6 +109,7 @@ module Cns
114
109
  )
115
110
  end
116
111
 
112
+ # @return [Boolean] mostra todas/novas transacoes
117
113
  def show_all?
118
114
  ops[:t] || false
119
115
  end
@@ -154,37 +150,42 @@ module Cns
154
150
  # @param [Array<Hash>] hlx ledger bitcoinde
155
151
  # @return [Array<Hash>] transaccao filtrada
156
152
  def pdel(hlx)
157
- hlx.map { |t| pdes(:time, t).merge(qtd: (t[:tp] == 'withdrawal' ? -1 : 1) * t[:qt].to_d, nxid: t[:txid], fee: t[:fee].to_d, moe: t[:moe].upcase) }
153
+ hlx.map { |t| pdes(:time, t) }
154
+ end
155
+
156
+ # @return [Hash] dados exchange bitcoinde - saldos & trades & deposits & withdrawals
157
+ def ded
158
+ @ded ||= {sl: pdea(api.account_de), tt: pdet(api.trades_de), tl: pdel(api.deposits_de + api.withdrawals_de)}
158
159
  end
159
160
 
160
161
  # @return [Array<String>] indices trades bigquery
161
162
  def bqkyt
162
- @bqkyt ||= show_all? ? [] : (bqd[:nt]&.map { |t| t[:txid] } || [])
163
+ @bqkyt ||= show_all? ? [] : bqd[:nt].map { |t| t[:txid] }
163
164
  end
164
165
 
165
166
  # @return [Array<Integer>] indices ledger bigquery
166
167
  def bqkyl
167
- @bqkyl ||= show_all? ? [] : (bqd[:nl]&.map { |l| l[:txid] } || [])
168
+ @bqkyl ||= show_all? ? [] : bqd[:nl].map { |l| l[:txid] }
168
169
  end
169
170
 
170
171
  # @return [Array<String>] lista txid trades novos
171
172
  def kyt
172
- @kyt ||= exd[:tt].map { |t| t[:trade_id] } - bqkyt
173
+ @kyt ||= ded[:tt].map { |t| t[:trade_id] } - bqkyt
173
174
  end
174
175
 
175
- # @return [Array<Integer>] lista txid ledger novos
176
+ # @return [Array<Integer>] lista nxid ledger novos
176
177
  def kyl
177
- @kyl ||= exd[:tl].map { |t| t[:txid] } - bqkyl
178
+ @kyl ||= ded[:tl].map { |t| t[:nxid] } - bqkyl
178
179
  end
179
180
 
180
- # @return [Array<Hash>] lista trades bitcoinde novos
181
+ # @return [Array<Hash>] lista trades novos bitcoinde
181
182
  def novcdet
182
- @novcdet ||= exd[:tt].select { |obj| kyt.include?(obj[:trade_id]) }
183
+ @novcdet ||= ded[:tt].select { |o| kyt.include?(o[:trade_id]) }
183
184
  end
184
185
 
185
- # @return [Array<Hash>] lista ledger (deposits + withdrawals) bitcoinde novos
186
+ # @return [Array<Hash>] lista ledgers (deposits + withdrawals) novos bitcoinde
186
187
  def novcdel
187
- @novcdel ||= exd[:tl].select { |obj| kyl.include?(obj[:txid]) }
188
+ @novcdel ||= ded[:tl].select { |o| kyl.include?(o[:nxid]) }
188
189
  end
189
190
  end
190
191
  end
data/lib/cns/etherscan.rb CHANGED
@@ -84,22 +84,22 @@ module Cns
84
84
 
85
85
  # mosta transacoes novas
86
86
  def mtransacoes_novas
87
- TT.each do |_, cfg|
88
- ntx = send(cfg[:new])
87
+ TT.each do |_, c|
88
+ ntx = send(c[:new])
89
89
  next unless ops[:v] && ntx.any?
90
90
 
91
- puts(cfg[:header])
92
- ntx.sort_by { |s| -s[cfg[:sork]] }.each { |t| puts(send(cfg[:format], t)) }
91
+ puts(c[:header])
92
+ ntx.sort_by { |s| -s[c[:sork]] }.each { |t| puts(send(c[:format], t)) }
93
93
  end
94
94
  end
95
95
 
96
96
  # mostra configuration text for adjusting days
97
97
  def mconfiguracao_ajuste_dias
98
- TT.each do |typ, cfg|
99
- ntx = send(cfg[:new])
98
+ TT.each do |p, c|
99
+ ntx = send(c[:new])
100
100
  next unless ntx.any?
101
101
 
102
- puts("\najuste dias transacoes #{typ}\n-h=#{ntx.sort_by { |s| -s[cfg[:sork]] }.map { |t| "#{t[cfg[:adjk]]}:0" }.join(' ')}")
102
+ puts("\najuste dias transacoes #{p}\n-h=#{ntx.sort_by { |s| -s[c[:sork]] }.map { |t| "#{t[c[:adjk]]}:0" }.join(' ')}")
103
103
  end
104
104
  end
105
105
 
@@ -226,44 +226,7 @@ module Cns
226
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))
227
227
  end
228
228
 
229
- # @param [Hash] abc account etherscan
230
- # @return [Hash] dados etherscan - address, saldo & transacoes
231
- def base_bc(abc)
232
- acc = abc[:account].downcase
233
- {
234
- ax: acc,
235
- sl: abc[:balance].to_d / (10**18),
236
- tx: ftik(acc, api.norml_es(acc)),
237
- ix: ftik(acc, api.inter_es(acc)),
238
- px: fppp(acc, api.block_es(acc)),
239
- wx: fwww(acc, api.withw_es(acc)),
240
- kx: ftik(acc, api.token_es(acc))
241
- }
242
- end
243
-
244
- # @param [Hash] wbq wallet bigquery
245
- # @param [Hash] hbc dados etherscan - address, saldo & transacoes
246
- # @return [Hash] dados juntos bigquery & etherscan
247
- def bq_bc(wbq, hbc)
248
- xbq = wbq[:ax]
249
- {
250
- id: wbq[:id],
251
- ax: xbq,
252
- bs: wbq[:sl],
253
- bt: bqd[:nt].select { |ont| ont[:iax].casecmp?(xbq) },
254
- bi: bqd[:ni].select { |oni| oni[:iax].casecmp?(xbq) },
255
- bp: bqd[:np].select { |onp| onp[:iax].casecmp?(xbq) },
256
- bw: bqd[:nw].select { |onw| onw[:iax].casecmp?(xbq) },
257
- bk: bqd[:nk].select { |onk| onk[:iax].casecmp?(xbq) },
258
- es: hbc[:sl],
259
- et: hbc[:tx],
260
- ei: hbc[:ix],
261
- ep: hbc[:px],
262
- ew: hbc[:wx],
263
- ek: hbc[:kx]
264
- }
265
- end
266
-
229
+ # @return [Boolean] mostra todas/novas transacoes
267
230
  def show_all?
268
231
  ops[:t] || false
269
232
  end
@@ -296,94 +259,132 @@ module Cns
296
259
  ary.map { |o| o.merge(itx: Integer(o[:withdrawalIndex]), iax: add, amount: o[:amount].to_d, timeStamp: Time.at(Integer(o[:timestamp]))) }
297
260
  end
298
261
 
262
+ # @param [Hash] aes account etherscan
263
+ # @return [Hash] dados etherscan - address, saldo & transacoes
264
+ def bses(aes)
265
+ acc = aes[:account].downcase
266
+ {
267
+ ax: acc,
268
+ sl: aes[:balance].to_d / (10**18),
269
+ tx: ftik(acc, api.norml_es(acc)),
270
+ ix: ftik(acc, api.inter_es(acc)),
271
+ px: fppp(acc, api.block_es(acc)),
272
+ wx: fwww(acc, api.withw_es(acc)),
273
+ kx: ftik(acc, api.token_es(acc))
274
+ }
275
+ end
276
+
277
+ # @param [Hash] wbq wallet bigquery
278
+ # @param [Hash] hes dados etherscan - address, saldo & transacoes
279
+ # @return [Hash] dados juntos bigquery & etherscan
280
+ def bqes(wbq, hes)
281
+ xbq = wbq[:ax]
282
+ {
283
+ id: wbq[:id],
284
+ ax: xbq,
285
+ bs: wbq[:sl],
286
+ bt: bqd[:nt].select { |t| t[:iax].casecmp?(xbq) },
287
+ bi: bqd[:ni].select { |i| i[:iax].casecmp?(xbq) },
288
+ bp: bqd[:np].select { |p| p[:iax].casecmp?(xbq) },
289
+ bw: bqd[:nw].select { |w| w[:iax].casecmp?(xbq) },
290
+ bk: bqd[:nk].select { |k| k[:iax].casecmp?(xbq) },
291
+ es: hes[:sl],
292
+ et: hes[:tx],
293
+ ei: hes[:ix],
294
+ ep: hes[:px],
295
+ ew: hes[:wx],
296
+ ek: hes[:kx]
297
+ }
298
+ end
299
+
299
300
  # @return [Array<String>] lista enderecos
300
301
  def lax
301
302
  @lax ||= bqd[:wb].map { |o| o[:ax] }
302
303
  end
303
304
 
304
305
  # @return [Array<Hash>] todos os dados etherscan - saldos & transacoes
305
- def bcd
306
- @bcd ||= api.account_es(lax).map { |o| base_bc(o) }
306
+ def esd
307
+ @esd ||= api.account_es(lax).map { |o| bses(o) }
307
308
  end
308
309
 
309
310
  # @return [Array<Hash>] todos os dados juntos bigquery & etherscan
310
311
  def dados
311
- @dados ||= bqd[:wb].map { |b| bq_bc(b, bcd.find { |e| b[:ax] == e[:ax] }) }
312
+ @dados ||= bqd[:wb].map { |b| bqes(b, esd.find { |e| b[:ax] == e[:ax] }) }
312
313
  end
313
314
 
314
315
  # @return [Array<Integer>] indices transacoes bigquery
315
316
  def bqidt
316
- @bqidt ||= show_all? ? [] : (bqd[:nt]&.map { |i| i[:itx] } || [])
317
+ @bqidt ||= show_all? ? [] : bqd[:nt].map { |i| i[:itx] }
317
318
  end
318
319
 
319
320
  # @return [Array<Integer>] indices transacoes bigquery
320
321
  def bqidi
321
- @bqidi ||= show_all? ? [] : (bqd[:ni]&.map { |i| i[:itx] } || [])
322
+ @bqidi ||= show_all? ? [] : bqd[:ni].map { |i| i[:itx] }
322
323
  end
323
324
 
324
325
  # @return [Array<Integer>] indices transacoes bigquery
325
326
  def bqidp
326
- @bqidp ||= show_all? ? [] : (bqd[:np]&.map { |i| i[:itx] } || [])
327
+ @bqidp ||= show_all? ? [] : bqd[:np].map { |i| i[:itx] }
327
328
  end
328
329
 
329
330
  # @return [Array<Integer>] indices transacoes bigquery
330
331
  def bqidw
331
- @bqidw ||= show_all? ? [] : (bqd[:nw]&.map { |i| i[:itx] } || [])
332
+ @bqidw ||= show_all? ? [] : bqd[:nw].map { |i| i[:itx] }
332
333
  end
333
334
 
334
335
  # @return [Array<Integer>] indices transacoes bigquery
335
336
  def bqidk
336
- @bqidk ||= show_all? ? [] : (bqd[:nk]&.map { |i| i[:itx] } || [])
337
+ @bqidk ||= show_all? ? [] : bqd[:nk].map { |i| i[:itx] }
337
338
  end
338
339
 
339
340
  # @return [Array<Integer>] indices transacoes novas (etherscan - bigquery)
340
341
  def idt
341
- @idt ||= bcd.map { |o| o[:tx].map { |i| i[:itx] } }.flatten - bqidt
342
+ @idt ||= esd.map { |o| o[:tx].map { |i| i[:itx] } }.flatten - bqidt
342
343
  end
343
344
 
344
345
  # @return [Array<Integer>] indices transacoes novas (etherscan - bigquery)
345
346
  def idi
346
- @idi ||= bcd.map { |o| o[:ix].map { |i| i[:itx] } }.flatten - bqidi
347
+ @idi ||= esd.map { |o| o[:ix].map { |i| i[:itx] } }.flatten - bqidi
347
348
  end
348
349
 
349
350
  # @return [Array<Integer>] indices transacoes novas (etherscan - bigquery)
350
351
  def idp
351
- @idp ||= bcd.map { |o| o[:px].map { |i| i[:itx] } }.flatten - bqidp
352
+ @idp ||= esd.map { |o| o[:px].map { |i| i[:itx] } }.flatten - bqidp
352
353
  end
353
354
 
354
355
  # @return [Array<Integer>] indices transacoes novas (etherscan - bigquery)
355
356
  def idw
356
- @idw ||= bcd.map { |o| o[:wx].map { |i| i[:itx] } }.flatten - bqidw
357
+ @idw ||= esd.map { |o| o[:wx].map { |i| i[:itx] } }.flatten - bqidw
357
358
  end
358
359
 
359
360
  # @return [Array<Integer>] indices transacoes novas (etherscan - bigquery)
360
361
  def idk
361
- @idk ||= bcd.map { |o| o[:kx].map { |i| i[:itx] } }.flatten - bqidk
362
+ @idk ||= esd.map { |o| o[:kx].map { |i| i[:itx] } }.flatten - bqidk
362
363
  end
363
364
 
364
365
  # @return [Array<Hash>] lista transacoes normais novas
365
366
  def novnetht
366
- @novnetht ||= bcd.map { |o| o[:tx].select { |t| idt.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
367
+ @novnetht ||= esd.map { |o| o[:tx].select { |t| idt.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
367
368
  end
368
369
 
369
370
  # @return [Array<Hash>] lista transacoes internas novas
370
371
  def novnethi
371
- @novnethi ||= bcd.map { |o| o[:ix].select { |t| idi.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
372
+ @novnethi ||= esd.map { |o| o[:ix].select { |t| idi.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
372
373
  end
373
374
 
374
375
  # @return [Array<Hash>] lista transacoes block novas
375
376
  def novnethp
376
- @novnethp ||= bcd.map { |o| o[:px].select { |t| idp.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
377
+ @novnethp ||= esd.map { |o| o[:px].select { |t| idp.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
377
378
  end
378
379
 
379
380
  # @return [Array<Hash>] lista transacoes withdrawals novas
380
381
  def novnethw
381
- @novnethw ||= bcd.map { |o| o[:wx].select { |t| idw.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
382
+ @novnethw ||= esd.map { |o| o[:wx].select { |t| idw.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
382
383
  end
383
384
 
384
385
  # @return [Array<Hash>] lista transacoes token novas
385
386
  def novnethk
386
- @novnethk ||= bcd.map { |o| o[:kx].select { |t| idk.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
387
+ @novnethk ||= esd.map { |o| o[:kx].select { |t| idk.include?(t[:itx]) } }.flatten.uniq { |i| i[:itx] }
387
388
  end
388
389
  end
389
390
  end
data/lib/cns/greymass.rb CHANGED
@@ -95,28 +95,7 @@ module Cns
95
95
  )
96
96
  end
97
97
 
98
- # @param [Hash] wbq wallet bigquery
99
- # @return [Hash] dados greymass - address, saldo & transacoes
100
- def base_bc(wbq)
101
- xbq = wbq[:ax]
102
- {ax: xbq, sl: peosa(xbq).reduce(:+), tx: peost(xbq, api.ledger_gm(xbq))}
103
- end
104
-
105
- # @param wbq (see base_bc)
106
- # @param [Hash] hbc dados greymass - address, saldo & transacoes
107
- # @return [Hash] dados juntos bigquery & greymass
108
- def bq_bc(wbq, hbc)
109
- xbq = wbq[:ax]
110
- {
111
- id: wbq[:id],
112
- ax: xbq,
113
- bs: wbq[:sl],
114
- bt: bqd[:nt].select { |o| o[:iax] == xbq },
115
- es: hbc[:sl],
116
- et: hbc[:tx]
117
- }
118
- end
119
-
98
+ # @return [Boolean] mostra todas/novas transacoes
120
99
  def show_all?
121
100
  ops[:t] || false
122
101
  end
@@ -133,11 +112,11 @@ module Cns
133
112
  # @param [Array<Hash>] ary lista transacoes
134
113
  # @return [Array<Hash>] lista transacoes filtrada
135
114
  def peost(add, ary)
136
- ary.map do |omp|
137
- act = omp[:action_trace][:act]
115
+ ary.map do |t|
116
+ act = t[:action_trace][:act]
138
117
  adt = act[:data]
139
118
  qtd = adt[:quantity].to_s
140
- omp.merge(
119
+ t.merge(
141
120
  name: act[:name],
142
121
  from: adt[:from],
143
122
  quantity: qtd.to_d,
@@ -145,36 +124,58 @@ module Cns
145
124
  to: adt[:to],
146
125
  memo: String(adt[:memo]).gsub(/\p{C}/, ''), # remove Non-Printable Characters
147
126
  moe: qtd[/[[:upper:]]+/],
148
- itx: omp[:global_action_seq],
127
+ itx: t[:global_action_seq],
149
128
  iax: add,
150
- block_time: Time.parse(omp[:block_time])
129
+ block_time: Time.parse(t[:block_time])
151
130
  )
152
131
  end
153
132
  end
154
133
 
134
+ # @param [Hash] wbq wallet bigquery
135
+ # @return [Hash] dados greymass - address, saldo & transacoes
136
+ def bsgm(wbq)
137
+ xbq = wbq[:ax]
138
+ {ax: xbq, sl: peosa(xbq).reduce(:+), tx: peost(xbq, api.ledger_gm(xbq))}
139
+ end
140
+
141
+ # @param wbq (see bsgm)
142
+ # @param [Hash] hgm dados greymass - address, saldo & transacoes
143
+ # @return [Hash] dados juntos bigquery & greymass
144
+ def bqgm(wbq, hgm)
145
+ xbq = wbq[:ax]
146
+ {
147
+ id: wbq[:id],
148
+ ax: xbq,
149
+ bs: wbq[:sl],
150
+ bt: bqd[:nt].select { |o| o[:iax] == xbq },
151
+ es: hgm[:sl],
152
+ et: hgm[:tx]
153
+ }
154
+ end
155
+
155
156
  # @return [Array<Hash>] todos os dados greymass - saldos & transacoes
156
- def bcd
157
- @bcd ||= bqd[:wb].map { |o| base_bc(o) }
157
+ def gmd
158
+ @gmd ||= bqd[:wb].map { |o| bsgm(o) }
158
159
  end
159
160
 
160
161
  # @return [Array<Hash>] todos os dados juntos bigquery & greymass
161
162
  def dados
162
- @dados ||= bqd[:wb].map { |b| bq_bc(b, bcd.find { |g| b[:ax] == g[:ax] }) }
163
+ @dados ||= bqd[:wb].map { |b| bqgm(b, gmd.find { |g| b[:ax] == g[:ax] }) }
163
164
  end
164
165
 
165
166
  # @return [Array<Integer>] indices transacoes bigquery
166
167
  def bqidt
167
- @bqidt ||= show_all? ? [] : (bqd[:nt]&.map { |i| i[:itx] } || [])
168
+ @bqidt ||= show_all? ? [] : bqd[:nt].map { |i| i[:itx] }
168
169
  end
169
170
 
170
171
  # @return [Array<Integer>] indices transacoes novas (greymass - bigquery)
171
172
  def idt
172
- @idt ||= bcd.map { |o| o[:tx].map { |i| i[:itx] } }.flatten - bqidt
173
+ @idt ||= gmd.map { |o| o[:tx].map { |i| i[:itx] } }.flatten - bqidt
173
174
  end
174
175
 
175
176
  # @return [Array<Hash>] lista transacoes novas
176
177
  def novneost
177
- @novneost ||= bcd.map { |obc| obc[:tx].select { |o| idt.include?(o[:itx]) } }.flatten
178
+ @novneost ||= gmd.map { |t| t[:tx].select { |o| idt.include?(o[:itx]) } }.flatten.uniq { |i| i[:itx] }
178
179
  end
179
180
  end
180
181
  end
data/lib/cns/kraken.rb CHANGED
@@ -23,31 +23,31 @@ module Cns
23
23
  @ops = pop.transform_keys(&:to_sym)
24
24
  end
25
25
 
26
- # @return [Hash] dados exchange kraken - saldos & transacoes trades e ledger
27
- def exd
28
- @exd ||= {sl: pusa(api.account_us), kt: pust(api.trades_us), kl: pusl(api.ledger_us)}
29
- end
30
-
31
- # @return [String] texto saldos & transacoes & ajuste dias
26
+ # mosta resumo saldos & transacoes & ajuste dias
32
27
  def mresumo
33
28
  puts("\nKRAKEN\ntipo kraken bigquery")
34
- exd[:sl].sort.each { |key, val| puts(fos(key, val)) }
29
+ usd[:sl].sort.each { |key, val| puts(fos(key, val)) }
35
30
  mtotais
36
31
 
37
32
  mtrades
38
33
  mledger
39
34
  return if novcust.empty?
40
35
 
41
- puts("\nstring ajuste dias dos trades\n-h=#{novcust.sort_by { |_, v| -v[:srx] }.map { |k, _v| "#{k}:0" }.join(' ')}")
36
+ puts("\nstring ajuste dias dos trades\n-h=#{novcust.sort_by { |i| -i[:srx] }.map { |o| "#{o[:txid]}:0" }.join(' ')}")
37
+ end
38
+
39
+ # @return [Hash] ledgers exchange kraken
40
+ def uskl
41
+ usd[:kl]
42
42
  end
43
43
 
44
44
  private
45
45
 
46
46
  # mosta contadores transacoes
47
47
  def mtotais
48
- vkt = exd[:kt].count
48
+ vkt = usd[:kt].count
49
49
  vnt = bqd[:nt].count
50
- vkl = exd[:kl].count
50
+ vkl = usd[:kl].count
51
51
  vnl = bqd[:nl].count
52
52
 
53
53
  puts("TRADES #{format('%<a>20i %<b>21i %<o>3.3s', a: vkt, b: vnt, o: vkt == vnt ? 'OK' : 'NOK')}")
@@ -59,7 +59,7 @@ module Cns
59
59
  return unless ops[:v] && novcust.any?
60
60
 
61
61
  puts("\ntrade data hora tipo par preco volume custo")
62
- novcust.sort_by { |_, v| -v[:srx] }.each { |k, t| puts(fot(k, t)) }
62
+ novcust.sort_by { |i| -i[:srx] }.each { |o| puts(fot(o)) }
63
63
  end
64
64
 
65
65
  # mosta transacoes ledger
@@ -67,14 +67,14 @@ module Cns
67
67
  return unless ops[:v] && novcusl.any?
68
68
 
69
69
  puts("\nledger data hora tipo moeda quantidade custo")
70
- novcusl.sort_by { |_, v| -v[:srx] }.each { |k, t| puts(fol(k, t)) }
70
+ novcusl.sort_by { |i| -i[:srx] }.each { |o| puts(fol(o)) }
71
71
  end
72
72
 
73
73
  # @param [String] moe codigo kraken da moeda
74
74
  # @param [BigDecimal] sal saldo kraken da moeda
75
75
  # @return [String] texto formatado saldos
76
76
  def fos(moe, sal)
77
- vbq = bqd[:sl][moe.downcase.to_sym].to_d
77
+ vbq = (bqd[:sl]&.fetch(moe.downcase.to_sym, nil) || 0).to_d
78
78
  format(
79
79
  '%<mo>-5.5s %<kr>21.9f %<bq>21.9f %<ok>3.3s',
80
80
  mo: moe.upcase,
@@ -84,13 +84,12 @@ module Cns
84
84
  )
85
85
  end
86
86
 
87
- # @param [Symbol] idx id da transacao
88
87
  # @param [Hash] htn trades kraken
89
88
  # @return [String] texto formatado trade
90
- def fot(idx, htx)
89
+ def fot(htx)
91
90
  format(
92
91
  '%<ky>-6.6s %<dt>19.19s %<ty>-10.10s %<mo>-8.8s %<pr>8.2f %<vl>10.4f %<co>13.2f',
93
- ky: idx,
92
+ ky: htx[:txid],
94
93
  dt: htx[:time].strftime('%F %T'),
95
94
  ty: "#{htx[:type]}/#{htx[:ordertype]}",
96
95
  mo: htx[:pair],
@@ -100,13 +99,12 @@ module Cns
100
99
  )
101
100
  end
102
101
 
103
- # @param idx (see fot)
104
102
  # @param [Hash] hln ledger kraken
105
103
  # @return [String] texto formatado ledger
106
- def fol(idx, hlx)
104
+ def fol(hlx)
107
105
  format(
108
106
  '%<ky>-6.6s %<dt>19.19s %<ty>-10.10s %<mo>-4.4s %<pr>18.7f %<vl>18.7f',
109
- ky: idx,
107
+ ky: hlx[:txid],
110
108
  dt: hlx[:time].strftime('%F %T'),
111
109
  ty: hlx[:type],
112
110
  mo: hlx[:asset],
@@ -115,6 +113,7 @@ module Cns
115
113
  )
116
114
  end
117
115
 
116
+ # @return [Boolean] mostra todas/novas transacoes
118
117
  def show_all?
119
118
  ops[:t] || false
120
119
  end
@@ -125,55 +124,51 @@ module Cns
125
124
  itm.select { |k, _| EM.include?(k) }.transform_values { |v| v.to_d }
126
125
  end
127
126
 
128
- # @param [Hash] itm transacao kraken
129
- # @return [Hash] transaccao filtrada
130
- def pusk(itm)
131
- itm.map do |k, v|
132
- t = Integer(v[:time])
133
- [k, v.merge(txid: k.to_s, srx: t, time: Time.at(t))]
134
- end.to_h
135
- end
136
-
137
- # @param [Hash] htx trade kraken
138
- # @return [Hash] transaccao filtrada
127
+ # @param [Array<Hash>] htx trades kraken
128
+ # @return [Array<Hash>] transaccoes filtradas
139
129
  def pust(htx)
140
- pusk(htx).transform_values { |t| t.merge(pair: t[:pair].upcase, price: t[:price].to_d, vol: t[:vol].to_d, cost: t[:cost].to_d) }
130
+ htx.map { |t| t.merge(pair: t[:pair].upcase, price: t[:price].to_d, vol: t[:vol].to_d, cost: t[:cost].to_d) }
141
131
  end
142
132
 
143
- # @param [Hash] hlx ledger kraken
144
- # @return [Hash] transaccao filtrada
133
+ # @param [Array<Hash>] hlx ledgers kraken
134
+ # @return [Array<Hash>] transaccoes filtradas
145
135
  def pusl(hlx)
146
- pusk(hlx).transform_values { |t| t.merge(asset: t[:asset].upcase, amount: t[:amount].to_d, fee: t[:fee].to_d) }
136
+ hlx.map { |t| t.merge(asset: t[:asset].upcase, amount: t[:amount].to_d, fee: t[:fee].to_d) }
137
+ end
138
+
139
+ # @return [Hash] dados exchange kraken - saldos & transacoes trades e ledger
140
+ def usd
141
+ @usd ||= {sl: pusa(api.account_us), kt: pust(api.trades_us), kl: pusl(api.ledger_us)}
147
142
  end
148
143
 
149
- # @return [Array<Symbol>] indices trades bigquery
144
+ # @return [Array<String>] indices trades bigquery
150
145
  def bqkyt
151
- @bqkyt ||= show_all? ? [] : (bqd[:nt]&.map { |t| t[:txid].to_sym } || [])
146
+ @bqkyt ||= show_all? ? [] : bqd[:nt].map { |t| t[:txid] }
152
147
  end
153
148
 
154
- # @return [Array<Symbol>] indices ledger bigquery
149
+ # @return [Array<String>] indices ledger bigquery
155
150
  def bqkyl
156
- @bqkyl ||= show_all? ? [] : (bqd[:nl]&.map { |l| l[:txid].to_sym } || [])
151
+ @bqkyl ||= show_all? ? [] : bqd[:nl].map { |l| l[:txid] }
157
152
  end
158
153
 
159
- # @return [Array<Symbol>] lista txid trades novos
154
+ # @return [Array<String>] lista txid trades novos
160
155
  def kyt
161
- @kyt ||= exd[:kt].keys - bqkyt
156
+ @kyt ||= usd[:kt].map { |t| t[:txid] } - bqkyt
162
157
  end
163
158
 
164
- # @return [Array<Symbol>] lista txid ledger novos
159
+ # @return [Array<String>] lista txid ledger novos
165
160
  def kyl
166
- @kyl ||= exd[:kl].keys - bqkyl
161
+ @kyl ||= usd[:kl].map { |t| t[:txid] } - bqkyl
167
162
  end
168
163
 
169
- # @return [Hash] trades kraken novos
164
+ # @return [Array<Hash>] trades novos kraken
170
165
  def novcust
171
- @novcust ||= exd[:kt].slice(*kyt)
166
+ @novcust ||= usd[:kt].select { |o| kyt.include?(o[:txid]) }
172
167
  end
173
168
 
174
- # @return [Hash] ledger kraken novos
169
+ # @return [Array<Hash>] ledgers novos kraken
175
170
  def novcusl
176
- @novcusl ||= exd[:kl].slice(*kyl)
171
+ @novcusl ||= usd[:kl].select { |o| kyl.include?(o[:txid]) }
177
172
  end
178
173
  end
179
174
  end
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.3'
4
+ VERSION = '0.9.5'
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.3
4
+ version: 0.9.5
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-04 00:00:00.000000000 Z
10
+ date: 2025-03-06 00:00:00.000000000 Z
11
11
  dependencies:
12
12
  - !ruby/object:Gem::Dependency
13
13
  name: bundler