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 +4 -4
- data/Gemfile.lock +3 -3
- data/lib/cns/apibc.rb +74 -59
- data/lib/cns/apice.rb +78 -35
- data/lib/cns/bigquery.rb +43 -20
- data/lib/cns/bitcoinde.rb +20 -19
- data/lib/cns/etherscan.rb +64 -63
- data/lib/cns/greymass.rb +34 -33
- data/lib/cns/kraken.rb +41 -46
- data/lib/cns/version.rb +1 -1
- metadata +2 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 5e6f05fc2c2fc0c7c1e34cda63d18338f37db37957be5f1af27c60041cd761b5
|
4
|
+
data.tar.gz: bab915764960144e4b713348958b173524f81ba42eb53d2d0df1242c2721d194
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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.
|
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.
|
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.
|
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
|
-
@
|
15
|
-
@
|
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
|
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
|
-
|
23
|
-
|
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
|
-
|
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
|
-
|
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
|
-
|
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
|
-
|
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
|
-
|
59
|
+
pag_es_req('tokentx', address)
|
59
60
|
end
|
60
61
|
|
61
62
|
# Get EOS account information
|
62
|
-
# @param
|
63
|
+
# @param [String] address EOS account name
|
63
64
|
# @return [Hash] Account details with resources
|
64
65
|
def account_gm(address)
|
65
|
-
|
66
|
-
|
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
|
-
|
74
|
+
trx = []
|
74
75
|
pos = 0
|
75
76
|
loop do
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
break if
|
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
|
-
|
84
|
+
trx
|
84
85
|
end
|
85
86
|
|
86
87
|
private
|
87
88
|
|
88
|
-
#
|
89
|
-
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
-
|
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
|
-
#
|
108
|
-
# @param
|
109
|
-
# @param
|
110
|
-
# @param
|
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
|
113
|
-
|
114
|
-
|
107
|
+
def pag_es_req(act, add, prm = {})
|
108
|
+
prm = prm.merge(offset: ESPS)
|
109
|
+
trx = []
|
110
|
+
pag = 1
|
115
111
|
loop do
|
116
|
-
|
117
|
-
break unless
|
112
|
+
res = es_req(act, add, pag, prm)
|
113
|
+
break unless res[:status] == '1'
|
118
114
|
|
119
|
-
|
120
|
-
|
121
|
-
break if
|
115
|
+
bth = res[:result] || []
|
116
|
+
trx.concat(bth)
|
117
|
+
break if bth.size < ESPS
|
122
118
|
|
123
|
-
|
119
|
+
pag += 1
|
124
120
|
end
|
125
|
-
|
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
|
-
#
|
134
|
-
|
135
|
-
|
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
|
-
#
|
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:
|
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
|
-
|
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:
|
86
|
-
|
87
|
-
break if
|
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
|
-
|
90
|
-
ofs +=
|
115
|
+
ary.concat(bth)
|
116
|
+
ofs += bth.size
|
91
117
|
end
|
92
|
-
|
118
|
+
ary
|
93
119
|
end
|
94
120
|
|
95
121
|
# Generic paginated request handler for Bitcoin.de
|
96
|
-
|
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 = "#{
|
131
|
+
url = "#{uri}?#{URI.encode_www_form(prm.merge(page: pag))}"
|
101
132
|
run_curl(@curl, url, headers: hde(url))
|
102
|
-
|
103
|
-
|
104
|
-
ary.concat(block_given? ? yield(
|
105
|
-
break if
|
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
|
-
|
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
|
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
|
-
#
|
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]),
|
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
|
-
|
145
|
-
fee: has[:network_fee],
|
146
|
-
|
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: '
|
191
|
+
moe: 'BTC'
|
149
192
|
}
|
150
193
|
end
|
151
194
|
|
152
|
-
#
|
153
|
-
# @param [
|
154
|
-
# @
|
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
|
-
|
157
|
-
|
158
|
-
|
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]
|
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(
|
170
|
-
{'api-key'
|
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
|
13
|
-
hetht: %i[blockNumber timeStamp
|
14
|
-
nethi: %w[blocknumber timestamp
|
15
|
-
hethi: %i[blockNumber timeStamp
|
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
|
21
|
-
hethk: %i[blockNumber timeStamp
|
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
|
-
|
158
|
-
|
159
|
-
np: sql("SELECT * FROM #{BD}.#{prx}p"),
|
160
|
-
|
161
|
-
|
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
|
-
|
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 |
|
201
|
-
novx = src.send("nov#{
|
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(
|
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], :
|
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.
|
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
|
-
|
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 =
|
43
|
+
vtt = ded[:tt].count
|
49
44
|
vnt = bqd[:nt].count
|
50
|
-
vtl =
|
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[:
|
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)
|
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? ? [] :
|
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? ? [] :
|
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 ||=
|
173
|
+
@kyt ||= ded[:tt].map { |t| t[:trade_id] } - bqkyt
|
173
174
|
end
|
174
175
|
|
175
|
-
# @return [Array<Integer>] lista
|
176
|
+
# @return [Array<Integer>] lista nxid ledger novos
|
176
177
|
def kyl
|
177
|
-
@kyl ||=
|
178
|
+
@kyl ||= ded[:tl].map { |t| t[:nxid] } - bqkyl
|
178
179
|
end
|
179
180
|
|
180
|
-
# @return [Array<Hash>] lista trades bitcoinde
|
181
|
+
# @return [Array<Hash>] lista trades novos bitcoinde
|
181
182
|
def novcdet
|
182
|
-
@novcdet ||=
|
183
|
+
@novcdet ||= ded[:tt].select { |o| kyt.include?(o[:trade_id]) }
|
183
184
|
end
|
184
185
|
|
185
|
-
# @return [Array<Hash>] lista
|
186
|
+
# @return [Array<Hash>] lista ledgers (deposits + withdrawals) novos bitcoinde
|
186
187
|
def novcdel
|
187
|
-
@novcdel ||=
|
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 |_,
|
88
|
-
ntx = send(
|
87
|
+
TT.each do |_, c|
|
88
|
+
ntx = send(c[:new])
|
89
89
|
next unless ops[:v] && ntx.any?
|
90
90
|
|
91
|
-
puts(
|
92
|
-
ntx.sort_by { |s| -s[
|
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 |
|
99
|
-
ntx = send(
|
98
|
+
TT.each do |p, c|
|
99
|
+
ntx = send(c[:new])
|
100
100
|
next unless ntx.any?
|
101
101
|
|
102
|
-
puts("\najuste dias transacoes #{
|
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
|
-
# @
|
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
|
306
|
-
@
|
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|
|
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? ? [] :
|
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? ? [] :
|
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? ? [] :
|
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? ? [] :
|
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? ? [] :
|
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 ||=
|
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 ||=
|
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 ||=
|
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 ||=
|
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 ||=
|
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 ||=
|
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 ||=
|
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 ||=
|
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 ||=
|
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 ||=
|
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
|
-
# @
|
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 |
|
137
|
-
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
|
-
|
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:
|
127
|
+
itx: t[:global_action_seq],
|
149
128
|
iax: add,
|
150
|
-
block_time: Time.parse(
|
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
|
157
|
-
@
|
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|
|
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? ? [] :
|
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 ||=
|
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 ||=
|
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
|
-
#
|
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
|
-
|
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 { |
|
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 =
|
48
|
+
vkt = usd[:kt].count
|
49
49
|
vnt = bqd[:nt].count
|
50
|
-
vkl =
|
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 { |
|
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 { |
|
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]
|
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(
|
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:
|
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(
|
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:
|
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]
|
129
|
-
# @return [Hash]
|
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
|
-
|
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
|
144
|
-
# @return [Hash]
|
133
|
+
# @param [Array<Hash>] hlx ledgers kraken
|
134
|
+
# @return [Array<Hash>] transaccoes filtradas
|
145
135
|
def pusl(hlx)
|
146
|
-
|
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<
|
144
|
+
# @return [Array<String>] indices trades bigquery
|
150
145
|
def bqkyt
|
151
|
-
@bqkyt ||= show_all? ? [] :
|
146
|
+
@bqkyt ||= show_all? ? [] : bqd[:nt].map { |t| t[:txid] }
|
152
147
|
end
|
153
148
|
|
154
|
-
# @return [Array<
|
149
|
+
# @return [Array<String>] indices ledger bigquery
|
155
150
|
def bqkyl
|
156
|
-
@bqkyl ||= show_all? ? [] :
|
151
|
+
@bqkyl ||= show_all? ? [] : bqd[:nl].map { |l| l[:txid] }
|
157
152
|
end
|
158
153
|
|
159
|
-
# @return [Array<
|
154
|
+
# @return [Array<String>] lista txid trades novos
|
160
155
|
def kyt
|
161
|
-
@kyt ||=
|
156
|
+
@kyt ||= usd[:kt].map { |t| t[:txid] } - bqkyt
|
162
157
|
end
|
163
158
|
|
164
|
-
# @return [Array<
|
159
|
+
# @return [Array<String>] lista txid ledger novos
|
165
160
|
def kyl
|
166
|
-
@kyl ||=
|
161
|
+
@kyl ||= usd[:kl].map { |t| t[:txid] } - bqkyl
|
167
162
|
end
|
168
163
|
|
169
|
-
# @return [Hash] trades kraken
|
164
|
+
# @return [Array<Hash>] trades novos kraken
|
170
165
|
def novcust
|
171
|
-
@novcust ||=
|
166
|
+
@novcust ||= usd[:kt].select { |o| kyt.include?(o[:txid]) }
|
172
167
|
end
|
173
168
|
|
174
|
-
# @return [Hash]
|
169
|
+
# @return [Array<Hash>] ledgers novos kraken
|
175
170
|
def novcusl
|
176
|
-
@novcusl ||=
|
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
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.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-
|
10
|
+
date: 2025-03-06 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: bundler
|