abank 0.6.8 → 0.7.1

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: 12bf681641a433a069f8657160ccb947e021b06ab94261f07a38da6001b4d970
4
- data.tar.gz: f807d26c094e3c6c2def749ef6c85e2aeeab848d07394b3f7c05c7f0e07b3757
3
+ metadata.gz: 5cdbf5848f71477ac0335fce39e8b98eec16ec5fbf0cce20d71d45100759779d
4
+ data.tar.gz: 0ca9a3cd18bbe22522695bb6de305bb238c5f9bf9166e8f3b1a8bf65258a58c3
5
5
  SHA512:
6
- metadata.gz: c643369008fc54e42f41ef77395c425d5dcf935ad3611bc83e2aba5120d604a9fc439d6aedeae7095426cb86a18d7e655270bdf74b9c4cc9d47be8ffa96b5e06
7
- data.tar.gz: 18bb5910ed4c3a40d57bb313eac62c5295d86d04c10f7b5fa1ba505af3e18e38100ec808302cd32d13467b032e637d42298b9bf48951b2c095df9cc16429ceff
6
+ metadata.gz: e613799e22925d9ab3af9ff414c6a5b66e8d37f4deb56c2a86cadd032a4ae62bb3fee6e4ec62c54490b9dd8f3626f54dd5407272aa9796354010b149e014acff
7
+ data.tar.gz: cb51cb9d37401a738ac8a83b294f24e48bc7a89ea06dc885d4b72ca5ddd09a05d521a049d878456e48880ffeb80303daf10c44eb0fa537e392e214667b69f4f1
data/.rubocop.yml CHANGED
@@ -18,7 +18,7 @@ Metrics/AbcSize:
18
18
 
19
19
  # Formatting Rules
20
20
  Layout/LineLength:
21
- Max: 160
21
+ Max: 185
22
22
  AllowHeredoc: true
23
23
  AllowURI: true
24
24
 
@@ -26,7 +26,7 @@ Layout/IndentationWidth:
26
26
  Width: 2
27
27
 
28
28
  Layout/SpaceInsideHashLiteralBraces:
29
- EnforcedStyle: space
29
+ EnforcedStyle: no_space
30
30
 
31
31
  Layout/SingleLineBlockChain:
32
32
  Enabled: false
@@ -65,3 +65,6 @@ Lint/AmbiguousOperatorPrecedence:
65
65
 
66
66
  Lint/ConstantResolution:
67
67
  Enabled: false
68
+
69
+ Lint/UnifiedInteger:
70
+ Enabled: false
data/Gemfile.lock CHANGED
@@ -1,7 +1,7 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- abank (0.6.8)
4
+ abank (0.7.1)
5
5
  google-cloud-bigquery
6
6
  roo
7
7
  thor
@@ -54,7 +54,7 @@ GEM
54
54
  logger
55
55
  faraday-net_http (3.4.0)
56
56
  net-http (>= 0.5.0)
57
- google-apis-bigquery_v2 (0.83.0)
57
+ google-apis-bigquery_v2 (0.84.0)
58
58
  google-apis-core (>= 0.15.0, < 2.a)
59
59
  google-apis-core (0.16.0)
60
60
  addressable (~> 2.5, >= 2.5.1)
@@ -64,7 +64,7 @@ GEM
64
64
  mutex_m
65
65
  representable (~> 3.0)
66
66
  retriable (>= 2.0, < 4.a)
67
- google-cloud-bigquery (1.51.1)
67
+ google-cloud-bigquery (1.52.0)
68
68
  bigdecimal (~> 3.0)
69
69
  concurrent-ruby (~> 1.0)
70
70
  google-apis-bigquery_v2 (~> 0.71)
@@ -72,12 +72,13 @@ GEM
72
72
  google-cloud-core (~> 1.6)
73
73
  googleauth (~> 1.9)
74
74
  mini_mime (~> 1.0)
75
- google-cloud-core (1.7.1)
75
+ google-cloud-core (1.8.0)
76
76
  google-cloud-env (>= 1.0, < 3.a)
77
77
  google-cloud-errors (~> 1.0)
78
- google-cloud-env (2.2.1)
78
+ google-cloud-env (2.2.2)
79
+ base64 (~> 0.2)
79
80
  faraday (>= 1.0, < 3.a)
80
- google-cloud-errors (1.4.0)
81
+ google-cloud-errors (1.5.0)
81
82
  google-logging-utils (0.1.0)
82
83
  googleauth (1.13.1)
83
84
  faraday (>= 1.0, < 3.a)
@@ -87,9 +88,10 @@ GEM
87
88
  multi_json (~> 1.11)
88
89
  os (>= 0.9, < 2.0)
89
90
  signet (>= 0.16, < 2.a)
90
- httpclient (2.8.3)
91
+ httpclient (2.9.0)
92
+ mutex_m
91
93
  jaro_winkler (1.6.0)
92
- json (2.10.1)
94
+ json (2.10.2)
93
95
  jwt (2.10.1)
94
96
  base64
95
97
  kramdown (2.5.1)
@@ -105,10 +107,10 @@ GEM
105
107
  mutex_m (0.3.0)
106
108
  net-http (0.6.0)
107
109
  uri
108
- nokogiri (1.18.2)
110
+ nokogiri (1.18.3)
109
111
  mini_portile2 (~> 2.8.2)
110
112
  racc (~> 1.4)
111
- nokogiri (1.18.2-x86_64-linux-gnu)
113
+ nokogiri (1.18.3-x86_64-linux-gnu)
112
114
  racc (~> 1.4)
113
115
  observer (0.1.2)
114
116
  os (1.1.4)
@@ -141,7 +143,7 @@ GEM
141
143
  roo (2.10.1)
142
144
  nokogiri (~> 1)
143
145
  rubyzip (>= 1.3.0, < 3.0.0)
144
- rubocop (1.72.1)
146
+ rubocop (1.74.0)
145
147
  json (~> 2.3)
146
148
  language_server-protocol (~> 3.17.0.2)
147
149
  lint_roller (~> 1.1.0)
@@ -152,7 +154,7 @@ GEM
152
154
  rubocop-ast (>= 1.38.0, < 2.0)
153
155
  ruby-progressbar (~> 1.7)
154
156
  unicode-display_width (>= 2.4.0, < 4.0)
155
- rubocop-ast (1.38.0)
157
+ rubocop-ast (1.38.1)
156
158
  parser (>= 3.3.1.0)
157
159
  rubocop-rake (0.7.1)
158
160
  lint_roller (~> 1.1)
@@ -164,7 +166,7 @@ GEM
164
166
  faraday (>= 0.17.5, < 3.a)
165
167
  jwt (>= 1.5, < 3.0)
166
168
  multi_json (~> 1.10)
167
- solargraph (0.51.2)
169
+ solargraph (0.52.0)
168
170
  backport (~> 1.2)
169
171
  benchmark
170
172
  bundler (~> 2.0)
@@ -182,6 +184,7 @@ GEM
182
184
  thor (~> 1.0)
183
185
  tilt (~> 2.0)
184
186
  yard (~> 0.9, >= 0.9.24)
187
+ yard-solargraph (~> 0.1)
185
188
  thor (1.3.2)
186
189
  tilt (2.6.0)
187
190
  trailblazer-option (0.1.2)
@@ -189,8 +192,10 @@ GEM
189
192
  unicode-display_width (3.1.4)
190
193
  unicode-emoji (~> 4.0, >= 4.0.4)
191
194
  unicode-emoji (4.0.4)
192
- uri (1.0.2)
195
+ uri (1.0.3)
193
196
  yard (0.9.37)
197
+ yard-solargraph (0.1.0)
198
+ yard (~> 0.9)
194
199
  zeitwerk (2.6.18)
195
200
 
196
201
  PLATFORMS
@@ -209,4 +214,4 @@ DEPENDENCIES
209
214
  yard
210
215
 
211
216
  BUNDLED WITH
212
- 2.6.3
217
+ 2.6.5
data/lib/abank/big.rb CHANGED
@@ -3,201 +3,192 @@
3
3
  require('google/cloud/bigquery')
4
4
 
5
5
  module Abank
6
- BD = 'hernanilr.ab'
7
-
8
- # @see Big
9
6
  class Big
10
7
  DF = '%Y-%m-%d'
11
8
 
9
+ # @return [Google::Cloud::Bigquery::QueryJob] job bigquery
12
10
  # @return [Hash] opcoes trabalho
13
- attr_reader :opcao
14
-
15
11
  # @return [Google::Cloud::Bigquery::Data] resultado do Structured Query Language (SQL) no bigquery
16
- attr_reader :bqres
17
-
18
12
  # @return [Integer] numero linhas afetadas pelo Data Manipulation Language (DML) no bigquery
19
- attr_reader :bqnrs
20
-
21
13
  # @return [Integer] contrato arrendamento em tratamento
22
- attr_reader :ctpos
23
-
24
14
  # @return [Integer] movimento do contrato arrendamento em tratamento
25
- attr_reader :mvpos
15
+ attr_reader :job, :opcao, :bqres, :bqnrs, :ctpos, :mvpos
26
16
 
27
17
  # acesso a base dados abank no bigquery
28
- #
29
18
  # @param [Thor::CoreExt::HashWithIndifferentAccess] opcoes trabalho
30
19
  # @option opcoes [String] :k ('') movimentos a apagar (keysin.mv)
20
+ # @option opcoes [Integer] :n (0) conta apagar movimentos >3 outras (mv)
31
21
  # @option opcoes [String] :c ('') id contrato arrendamento (re)
32
22
  # @option opcoes [String] :d ('') data inicio contrato arrendamento (re)
33
23
  # @option opcoes [Boolean] :t (false) trabalha todas as rendas? (re)
34
24
  # @return [Hash] opcoes trabalho
35
25
  def initialize(opcoes = {})
36
26
  @opcao = opcoes
37
- opcao
38
27
  end
39
28
 
40
- # @return [Google::Cloud::Bigquery] API bigquery
41
- def bqapi
42
- @bqapi ||= Google::Cloud::Bigquery.new
43
- end
29
+ # insere & classifica movimentos no bigquery
30
+ # @return [Big] acesso a base dados abank no bigquery
31
+ def mv_insert
32
+ return self if mvvls.empty?
44
33
 
45
- # @return [String] movimentos a apagar (keysin.mv)
46
- def mvkys
47
- opcao[:k].to_s.scan(/[-+]?\d+/).join(',')
34
+ dml("insert #{BD}.mv VALUES#{mvvls.join(',')}")
35
+ puts("MOVIMENTOS INSERIDOS #{bqnrs}")
36
+ mv_classifica if bqnrs.positive?
37
+ self
48
38
  end
49
39
 
50
40
  # apaga movimentos & suas rendas associadas no bigquery
51
- #
52
41
  # @return [Big] acesso a base dados abank no bigquery
53
42
  def mv_delete
54
43
  @ctlct = []
55
- unless mvkys.empty?
56
- # obtem lista contratos arrendamento associados aos movimentos a apagar
57
- @ctlct = sql("select distinct ct from #{BD}.gmr where ky in(#{mvkys})")
58
-
59
- re_apaga.mv_delete_dml
60
- end
44
+ return self if mvkys.empty? && docnt.zero?
45
+
46
+ # obtem lista contratos arrendamento associados aos movimentos a apagar
47
+ @ctlct =
48
+ if docnt.zero?
49
+ sql("select distinct ct from #{BD}.gmr where ky IN UNNEST(@kys)", kys: mvkys)
50
+ else
51
+ sql("select distinct ct from #{BD}.gmr where nc=@nc", nc: docnt)
52
+ end
53
+ re_apaga
54
+ mv_delete_dml
61
55
  self
62
56
  end
63
57
 
64
- # apaga movimentos no bigquery
65
- def mv_delete_dml
66
- dml("delete from #{BD}.mv where #{BD}.ky(dl,dv,ds,vl,nc,ex) in(#{mvkys})")
67
- puts("MOVIMENTOS APAGADOS #{bqnrs}")
68
- end
69
-
70
- # (see CLI#tag)
58
+ # classifica movimentos no bigquery
59
+ # @return [Big] acesso a base dados abank no bigquery
71
60
  def mv_classifica
72
- @ctlct = []
61
+ @ctlct = sql("select * from #{BD}.gnr")
73
62
  stp("call #{BD}.uct()")
74
63
  puts("MOVIMENTOS CLASSIFICADOS #{bqnrs}")
75
- @ctlct = sql("select ct from #{BD}.ca") if bqnrs.positive?
76
64
  self
77
65
  end
78
66
 
79
- # (see CLI#criact)
67
+ # cria contrato arrendamento no bigquery
68
+ # @return [Big] acesso a base dados abank no bigquery
80
69
  def ct_cria
81
- unless ct_existe?
82
- dml("insert into #{BD}.re #{sql_contrato_mv}")
70
+ if sql("SELECT ct FROM #{BD}.ca WHERE ct=@ct", ct: opcao[:c]).empty?
71
+ if valid_dc?
72
+ dml("insert #{BD}.re select @ct,EXTRACT(YEAR FROM DATE(@dc)),0,DATE(@dc),0", ct: opcao[:c], dc: opcao[:d])
73
+ else
74
+ dml("insert #{BD}.re select ct,EXTRACT(YEAR FROM DATE_TRUNC(dl,MONTH)),0,DATE_TRUNC(dl,MONTH),0 from #{BD}.mv where ct=@ct order by dl limit 1", ct: opcao[:c])
75
+ end
83
76
  puts("CONTRATO #{opcao[:c]} #{bqnrs.zero? ? 'NAO ' : ''}INSERIDO")
77
+ else
78
+ @bqnrs = 1
79
+ puts('CONTRATO JA EXISTE')
84
80
  end
85
- # processar rendas sim/nao?
81
+ @ctlct = [{ct: opcao[:c]}]
86
82
  return unless bqnrs.positive? && opcao[:t]
87
83
 
88
84
  # processa rendas associadas ao contrato arrendamento
89
85
  ct_dados.re_insert
90
86
  end
91
87
 
92
- # @return [Boolean] contrato arrendamento ja existe sim/nao?
93
- def ct_existe?
94
- @ctlct = [{ ct: opcao[:c] }]
95
- vaz = sql("select ct from #{BD}.ca where ct in(#{str_lc})").empty?
96
- unless vaz
97
- @bqnrs = 1
98
- puts('CONTRATO JA EXISTE')
99
- end
100
- !vaz
101
- end
102
-
103
88
  # (see CLI#apagact)
104
89
  def ct_apaga
105
- @ctlct = [{ ct: opcao[:c] }]
90
+ @ctlct = [{ct: opcao[:c]}]
106
91
  re_delete_dml
107
92
  end
108
93
 
109
94
  # optem lista dados contrato arrendamento (inclui lista movimentos novos)
110
- #
111
95
  # @return [Big] acesso a base dados abank no bigquery
112
96
  def ct_dados
113
97
  ctlct.map! do |ctr|
114
98
  opcao[:c] = ctr[:ct]
115
- lre = sql(sql_last_re).first
116
- lre[:dl] += -1 if lre[:cnt].zero?
117
- ctr.merge(lre, mv: sql(sql_novo_mv(lre[:dl])))
99
+ lre = sql("select * from #{BD}.glr where ct=@ct order by ano desc,cnt desc limit 1", ct: opcao[:c]).first
100
+ lre[:dl] -= 1 if lre[:cnt].zero?
101
+ ctr.merge(lre, mv: sql("select * from #{BD}.gmr where ct=@ct and dv>=@ud order by 1,2", ct: opcao[:c], ud: lre[:dl] + 1))
118
102
  end
119
103
  self
120
104
  end
121
105
 
122
- # @param [Array] are lista rendas novas atual
123
- # @return [Array<String>] lista rendas novas duma lista contratos arrendamento
124
- def ct_rendas(lre = [])
125
- while ctpos < ctlct.size
126
- @mvpos = 0
127
- lre += re_rendas
128
- @ctpos += 1
106
+ # insere rendas associadas a lista contratos arrendamento no bigquery
107
+ def re_insert
108
+ @ctpos = 0
109
+ vls = ct_rendas.join(',')
110
+ if vls.empty?
111
+ puts('NAO EXISTEM RENDAS NOVAS')
112
+ else
113
+ dml("insert #{BD}.re VALUES#{vls}")
114
+ puts("RENDAS #{cta.join(',')} CRIADAS #{bqnrs}")
129
115
  end
130
- lre
131
116
  end
132
117
 
133
- # @example sem dados movimentos
134
- # [{ ct: 'r03000' }, ...]
135
- # @example com dados movimentos
136
- # [{ct: 'r03000', dc: '2020-03-01', ano: 2020, cnt: 0, dl: '2020-03-01', mv: [{dl: '2020-03-02', vl: 30}, ...] }]
137
- # @return [Array<Hash>] lista dados contrato arrendamento (inclui lista movimentos novos)
138
- def ctlct
139
- @ctlct ||= []
118
+ # @return [String] sql inicio contrato arrendamento sem movimentos
119
+ # (see CLI#recriare)
120
+ def re_atualiza
121
+ # [re]cria rendas novas/todas dos contratos ativos
122
+ @ctlct = sql("select ct from #{BD}.ca")
123
+ re_apaga.ct_dados.re_insert
140
124
  end
141
125
 
142
- # @return [String] texto formatado que representa lista de contratos arrendamento
143
- def str_lc(sep = "'")
144
- ctlct.map { |cid| sep + cid[:ct] + sep }.join(',')
145
- end
126
+ private
146
127
 
147
- # @return [String] sql para obter ultima renda do contrato arrendamento
148
- def sql_last_re
149
- "select * from #{BD}.glr where ct='#{opcao[:c]}' order by ano desc,cnt desc limit 1"
128
+ # @return [Integer] numero conta movimentos apagar
129
+ def docnt
130
+ @docnt ||= opcao[:n] > 3 ? opcao[:n] : 0
150
131
  end
151
132
 
152
- # @return [String] sql para obter movimentos novos (depois da ultima renda do contrato arrendamento)
153
- def sql_novo_mv(mdl)
154
- "select * from #{BD}.gmn where ct='#{opcao[:c]}' and dl>='#{(mdl + 1).strftime(DF)}' order by 1,2"
133
+ # @return [Google::Cloud::Bigquery] API bigquery
134
+ def bqapi
135
+ @bqapi ||= Google::Cloud::Bigquery.new
155
136
  end
156
137
 
157
- # @return [String] sql para obter dados do inicio contrato arrendamento
158
- def sql_contrato_mv
159
- cti = opcao[:c]
160
- dat = opcao[:d]
161
- if dat.empty?
162
- 'select ct,EXTRACT(YEAR FROM DATE_TRUNC(dl,MONTH)) ano,0 cnt,DATE_TRUNC(dl,MONTH) dl,0 dias ' \
163
- "from #{BD}.mv where ct='#{cti}' order by dl limit 1"
164
- else
165
- "select '#{cti}' ct,EXTRACT(YEAR FROM DATE '#{dat}') ano,0 cnt,DATE '#{dat}' dl,0 dias"
166
- end
138
+ # @return [String] movimentos a apagar (keysin.mv)
139
+ def mvkys
140
+ @mvkys ||= opcao[:k].to_s.scan(/[-+]?\d+/).map(&:to_i)
167
141
  end
168
142
 
169
- # (see CLI#recriare)
170
- def re_atualiza
171
- # [re]cria rendas novas/todas dos contratos ativos
172
- @ctlct = sql("select ct from #{BD}.ca")
173
- re_apaga.ct_dados.re_insert
174
- end
143
+ # @return [Boolean] data contrato arrendamento valida?
144
+ def valid_dc?
145
+ s = opcao[:d].to_s.strip
146
+ return false unless s.length.positive?
175
147
 
176
- # insere rendas associadas a lista contratos arrendamento no bigquery
177
- def re_insert
178
- @ctpos = 0
179
- vls = ct_rendas.join(',')
180
- if vls.empty?
181
- puts('NAO EXISTEM RENDAS NOVAS')
182
- else
183
- dml("insert #{BD}.re VALUES#{vls}")
184
- puts("RENDAS #{str_lc('')} CRIADAS #{bqnrs}")
185
- end
148
+ d = Date.parse(s)
149
+ opcao[:d] = d.iso8601
150
+ Date.valid_date?(d.year, d.month, d.day)
151
+ rescue StandardError
152
+ false
186
153
  end
187
154
 
188
155
  # apaga rendas da lista de contrato arrendamento
189
- #
190
156
  # @return [Big] acesso a base dados abank no bigquery
191
157
  def re_apaga
192
158
  return self if ctlct.empty?
193
159
 
194
160
  # para nao apagar contrato arrendamento - somente as rendas
195
161
  opcao[:t] = false
196
-
197
162
  re_delete_dml
198
163
  self
199
164
  end
200
165
 
166
+ # @example sem dados movimentos
167
+ # [{ ct: 'r03000' }, ...]
168
+ # @example com dados movimentos
169
+ # [{ct: 'r03000', dc: '2020-03-01', ano: 2020, cnt: 0, dl: '2020-03-01', mv: [{dl: '2020-03-02', vl: 30}, ...] }]
170
+ # @return [Array<Hash>] lista dados contrato arrendamento (inclui lista movimentos novos)
171
+ def ctlct
172
+ @ctlct ||= []
173
+ end
174
+
175
+ # @return [Array<String>] lista de contratos arrendamento
176
+ def cta
177
+ ctlct.map { |c| c[:ct] }
178
+ end
179
+
180
+ # @param [Array] are lista rendas novas atual
181
+ # @return [Array<String>] lista rendas novas duma lista contratos arrendamento
182
+ def ct_rendas
183
+ lre = []
184
+ ctlct.each do
185
+ @mvpos = 0
186
+ lre += re_rendas
187
+ @ctpos += 1
188
+ end
189
+ lre
190
+ end
191
+
201
192
  # @return [Array<String>] lista rendas novas dum contrato arrendamento
202
193
  def re_rendas
203
194
  lre = []
@@ -211,21 +202,31 @@ module Abank
211
202
  # @return [String] renda formatada (values.re)
212
203
  def re_nova_renda
213
204
  re_proximos_dados
214
- "('#{re_atual[:ct]}',#{ano},#{cnt},'#{re_atual_mv[:dl].strftime(DF)}',#{dias})"
205
+ "('#{re_atual[:ct]}',#{ano},#{cnt},'#{re_atual_mv[:dl].iso8601}',#{dias})"
215
206
  end
216
207
 
217
208
  # @return [Hash] dados contrato arrendamento (inclui lista movimentos novos)
218
209
  def re_proximos_dados
219
210
  # valor renda paga retirada do movimento
220
211
  re_atual_mv[:vl] -= re_atual[:vr]
221
- dre = cnt.zero? ? Date.new(re_atual[:dc].year, re_atual[:dc].month, 1) : Date.new(ano, cnt, 1) >> 1
212
+ dre = cnt.zero? ? Date.new(re_atual[:dc].year, re_atual[:dc].month, 1) : Date.new(ano, cnt, 1).next_month
222
213
  re_atual.merge!(ano: dre.year, cnt: dre.month)
223
214
  end
224
215
 
216
+ # apaga movimentos no bigquery
217
+ def mv_delete_dml
218
+ if docnt.zero?
219
+ dml("delete from #{BD}.mv where #{BD}.ky(dl,dv,ds,vl,nc,ex) IN UNNEST(@kys)", kys: mvkys)
220
+ else
221
+ dml("delete from #{BD}.mv where nc=@nc", nc: docnt)
222
+ end
223
+ puts("MOVIMENTOS APAGADOS #{bqnrs}")
224
+ end
225
+
225
226
  # apaga rendas da lista de contratos arrendamento
226
227
  def re_delete_dml
227
- dml("delete from #{BD}.re where ct in(#{str_lc})#{opcao[:t] ? '' : ' and cnt>0'}")
228
- puts("RENDAS #{str_lc('')} APAGADAS #{bqnrs}")
228
+ dml("delete from #{BD}.re where ct IN UNNEST(@cts)#{' and cnt>0' unless opcao[:t]}", cts: cta)
229
+ puts("RENDAS #{cta.join(',')} APAGADAS #{bqnrs}")
229
230
  end
230
231
 
231
232
  # @return [Boolean] movimento com saldo suficiente?
@@ -258,48 +259,41 @@ module Abank
258
259
  re_atual_mv[:dl].mjd - Date.new(ano, cnt, 1).mjd
259
260
  end
260
261
 
262
+ # cria job bigquery & verifica execucao
261
263
  # @param [String] cmd comando a executar
262
- # @return [Google::Cloud::Bigquery::QueryJob] tarefa SQL/DML no bigquery
263
- def job(cmd)
264
- bqjob = bqapi.query_job(cmd)
265
- bqjob.wait_until_done!
266
- puts(bqjob.error['message']) if bqjob.failed?
267
- bqjob
264
+ # @param [Hash] prm parâmetros para a query
265
+ # @return [Boolean] job ok?
266
+ def job?(cmd, prm = {})
267
+ @job = bqapi.query_job(cmd, params: prm, priority: 'BATCH')
268
+ job.wait_until_done!
269
+ return true unless job.failed?
270
+
271
+ puts("BigQuery: #{job.error['message']}\n#{cmd}")
272
+ false
268
273
  end
269
274
 
270
275
  # executa Structured Query Language (SQL) no bigquery
271
- #
272
- # @param (see job)
273
- # @param [Array] erro resultado quando falha execucao
276
+ # @param [String] cmd comando SQL a executar
277
+ # @param [Hash] prm parâmetros para a query
274
278
  # @return [Google::Cloud::Bigquery::Data] resultado do SQL
275
- def sql(cmd, erro = [])
276
- # se job.failed? executa job(cmd).data => StandardError
277
- @bqres = job(cmd).data
278
- rescue StandardError
279
- @bqres = erro
279
+ def sql(cmd, prm = {})
280
+ @bqres = job?(cmd, prm) ? job.data : []
280
281
  end
281
282
 
282
283
  # executa Data Manipulation Language (DML) no bigquery
283
- #
284
- # @param (see job)
285
- # @return [Integer] numero rows afetadas pelo DML
286
- def dml(cmd)
287
- # se job.failed? executa Integer(nil) => StandardError
288
- @bqnrs = Integer(job(cmd).num_dml_affected_rows)
289
- rescue StandardError
290
- @bqnrs = 0
284
+ # @param [String] cmd comando DML a executar
285
+ # @param [Hash] prm parâmetros para a query
286
+ # @return [Integer] numero linhas afetadas
287
+ def dml(cmd, prm = {})
288
+ @bqnrs = job?(cmd, prm) ? job.num_dml_affected_rows.to_i : 0
291
289
  end
292
290
 
293
- # executa Stored Procedure (STP) with DML operations no bigquery
294
- #
295
- # @param (see job)
291
+ # executa Stored Procedure (STP) no bigquery
292
+ # @param [String] cmd comando STP a executar
296
293
  # @return [Integer] numero rows afetadas pelo STP
297
294
  def stp(cmd)
298
295
  # last command STP=SELECT @@row_count AS rows_affected;
299
- # se job.failed? executa Integer(nil) => StandardError
300
- @bqnrs = Integer(job(cmd).data.first[:rows_affected])
301
- rescue StandardError
302
- @bqnrs = 0
296
+ @bqnrs = job?(cmd) ? job.data.first[:rows_affected].to_i : 0
303
297
  end
304
298
  end
305
299
  end
data/lib/abank/folha.rb CHANGED
@@ -3,16 +3,10 @@
3
3
  require('roo')
4
4
 
5
5
  module Abank
6
- # @see Folha
7
6
  class Folha < Big
8
- # @return [Roo::Excelx] folha calculo a processar
9
- # attr_reader :folha
10
-
11
7
  # @return [Array] row folha calculo em processamento
12
- attr_reader :rowfc
13
-
14
8
  # @return [String] movimentos a inserir (values.mv)
15
- attr_reader :mvvls
9
+ attr_reader :rowfc, :mvvls
16
10
 
17
11
  # acesso a folha calculo & base dados abank no bigquery
18
12
  #
@@ -21,150 +15,132 @@ module Abank
21
15
  # @option opcoes [Boolean] :s (false) apaga movimento similar? (mv)
22
16
  # @option opcoes [Boolean] :e (false) apaga movimento igual? (mv)
23
17
  # @option opcoes [Boolean] :i (false) insere movimento novo? (mv)
18
+ # @option opcoes [Integer] :n (0) conta dos movimentos (mv)
24
19
  # @option opcoes [String] :v ('') data valor movimentos (mv)
25
20
  # @option opcoes [String] :g ('') classificacao movimentos (mv)
26
21
  def initialize(opcoes = {})
27
- @opcao = super
28
- @opcao[:s] = opcoes.fetch(:s, false)
29
- @opcao[:e] = opcoes.fetch(:e, false)
30
- @opcao[:i] = opcoes.fetch(:i, false)
31
- @opcao[:v] = opcoes.fetch(:v, '')
32
- @opcao[:g] = opcoes.fetch(:g, '')
33
- # acumuladores necessitam init
34
- @opcao[:k] = ''
35
- @mvvls = ''
36
- end
37
-
38
- # @return [Roo::Excelx] folha calculo a processar
39
- def folha
40
- @folha ||= Roo::Spreadsheet.open(opcao[:f])
41
- rescue StandardError => e
42
- raise("Erro ao abrir a folha de cálculo: #{e.message}")
22
+ super
23
+ @opcao = opcao.merge(
24
+ s: opcoes.fetch(:s, false),
25
+ e: opcoes.fetch(:e, false),
26
+ i: opcoes.fetch(:i, false),
27
+ n: opcoes.fetch(:n, 0),
28
+ v: opcoes.fetch(:v, ''),
29
+ g: opcoes.fetch(:g, ''),
30
+ k: +'',
31
+ f: opcoes[:f]
32
+ )
33
+ @mvvls = []
43
34
  end
44
35
 
45
36
  # carrega/mostra folha calculo
46
37
  def processa_xls
47
38
  puts("\n#{folha.info}")
48
- folha.sheet(0).parse(header_search: ['Data Lanc.', 'Data Valor', 'Descrição', 'Valor']) do |row|
49
- puts(processa_linha) if ok?(row)
39
+ mvs = sql("select * from #{BD}.gmv where nc=@nc", nc: conta).group_by { |m| [m[:dl], m[:vl].to_f] }
40
+ folha.sheet(0).parse(header_search: ['Data Lanc.', 'Data Valor', 'Descrição', 'Valor']) do |r|
41
+ next unless valid?(r.values)
42
+
43
+ @bqres = mvs[[rowfc[0], rowfc[3]]] || []
44
+ if bqres.empty?
45
+ puts(lnexi)
46
+ elsif bqres.one? && bqres.first[:ds].strip == rowfc[2]
47
+ puts(lexis)
48
+ elsif bqres.one?
49
+ puts(lsiml)
50
+ else
51
+ puts(lmult)
52
+ end
50
53
  end
51
54
  return unless opcao[:i]
52
55
 
53
- # processa movimentos & atualiza rendas
56
+ # para nao apagar movimentos duma conta, por aqui somente com keys opcao[:k]
57
+ opcao[:n] = 0
54
58
  mv_delete.mv_insert.ct_dados.re_insert
55
59
  end
56
60
 
57
- # @return [Integer] numero conta associado a folha calculo
58
- # @example
59
- # mov*.xlsx --> 1 --> conta-corrente
60
- # movCard*.xlsx --> 2 --> conta-cartao
61
- def conta
62
- opcao[:f].match?(/card/i) ? 2 : 1
63
- end
64
-
65
- # @param [Hash] linha da folha calculo em processamento
66
- # @return [Boolean] linha com valores para processar?
67
- def ok?(linha)
68
- @rowfc = linha.values
69
- return false if rowfc.first.is_a?(String)
61
+ private
70
62
 
71
- rowfc[2] = rowfc[2].strip
72
- rowfc[3] = rowfc[3] * -1 if conta > 1
73
- true
63
+ # @return [Roo::Excelx] folha calculo a processar
64
+ def folha
65
+ @folha ||= Roo::Spreadsheet.open(opcao[:f])
66
+ rescue StandardError
67
+ raise("Erro ao abrir a folha de cálculo: #{opcao[:f]}")
74
68
  end
75
69
 
76
- # @return [String] texto informativo formatado da linha processada
77
- def processa_linha
78
- # pesquisa existencia linha folha calculo no bigquery
79
- # array.count = 0 ==> pode carregar esta linha
80
- # array.count = 1 ==> mais testes necessarios
81
- # array.count > 1 ==> nao pode carregar esta linha
82
- sql(sql_existe_mv, [{}, {}])
83
-
84
- return linha_base + values_mv if linha_naoexiste?
85
- return linha_existe if linha_existe?
86
- return linha_similar if linha_simila?
70
+ # @return [Integer] obter numero conta a partir das opcoes
71
+ def fconta
72
+ return opcao[:n] if opcao[:n] > 2
87
73
 
88
- linha_multiplas
74
+ opcao[:f].match?(/card/i) ? 2 : 1
89
75
  end
90
76
 
91
- # insere & classifica movimentos no bigquery
92
- #
93
- # @return [Big] acesso a base dados abank no bigquery
94
- def mv_insert
95
- unless mvvls.empty?
96
- @mvvls = mvvls[1..] if mvvls.first == ','
97
- dml("insert #{BD}.mv VALUES#{mvvls}")
98
- puts("MOVIMENTOS INSERIDOS #{bqnrs}")
99
- mv_classifica if bqnrs.positive?
100
- end
101
- self
77
+ # @example
78
+ # mov*.xlsx --> 1 --> conta-corrente
79
+ # movCard*.xlsx --> 2 --> conta-cartao
80
+ # opcao[:n] --> 3 --> conta-cash
81
+ # opcao[:n] --> n --> conta-outras
82
+ # @return [Integer] numero conta associado a folha calculo
83
+ def conta
84
+ @conta ||= fconta
102
85
  end
103
86
 
104
- # @return [Boolean] linha folha calculo nao existe no bigquery?
105
- def linha_naoexiste?
106
- bqres.empty?
107
- end
87
+ # @param [Array] row folha calculo em processamento
88
+ # @return [Boolean] linha com valores correctos para processar?
89
+ def valid?(row)
90
+ return false unless row[0].is_a?(Date) && row[1].is_a?(Date)
108
91
 
109
- # @return [Boolean] linha folha calculo existe no bigquery?
110
- def linha_existe?
111
- bqres.count == 1 && bqres.first[:ds].strip == rowfc[2]
92
+ row[2] = row[2].to_s.strip.gsub("'", '').gsub('\\', '') # Descrição
93
+ row[3] = row[3].to_f * (conta == 2 ? -1 : 1) # Valor
94
+ @rowfc = row
95
+ true
96
+ rescue StandardError => e
97
+ puts("Error processing row values: #{e.message}\nRow: #{row.inspect}")
98
+ false
112
99
  end
113
100
 
114
- # @return [Boolean] linha folha calculo existe parecida no bigquery?
115
- def linha_simila?
116
- bqres.count == 1 && bqres.first[:ds].strip != rowfc[2]
101
+ # @return [String] texto base formatado para display
102
+ def lbase
103
+ format('%<dt>10s %<v3>-34.34s %<v4>8.2f', dt: rowfc[0].strftime(DF), v3: rowfc[2], v4: rowfc[3])
117
104
  end
118
105
 
119
- # @return [String] texto base formatado para display
120
- def linha_base
121
- "#{rowfc.first.strftime(DF)} #{format('%<v3>-34.34s %<v4>8.2f', v3: rowfc[2], v4: rowfc[3])}"
106
+ # @return [String] novo texto base formatado para display
107
+ def lnexi
108
+ @mvvls << "('#{rowfc[0].iso8601}','#{dvc.iso8601}','#{rowfc[2]}',#{rowfc[3]},#{conta},#{dvc.year},#{dvc.month},'#{tpc}',#{ctc},null,null)"
109
+ "#{lbase} NOVO"
122
110
  end
123
111
 
124
112
  # @return [String] texto linha existente formatada para display
125
- def linha_existe
113
+ def lexis
126
114
  add_kys if opcao[:e]
127
- "#{linha_base} EXIS #{format('%<v1>20d', v1: bqres.first[:ky])}"
115
+ "#{lbase} EXIS #{format('%<v1>20d', v1: bqres.first[:ky])}"
128
116
  end
129
117
 
130
118
  # @return [String] texto linha similar formatada para display
131
- def linha_similar
119
+ def lsiml
132
120
  add_kys if opcao[:s]
133
- "#{linha_base} SIMI #{format('%<v1>20d %<v2>-34.34s', v1: bqres.first[:ky], v2: bqres.first[:ds].strip)}"
121
+ "#{lbase} SIMI #{format('%<v1>20d %<v2>-34.34s', v1: bqres.first[:ky], v2: bqres.first[:ds].strip)}"
134
122
  end
135
123
 
136
124
  # @return [String] texto linha existencia multipla formatada para display
137
- def linha_multiplas
138
- "#{linha_base} ML#{format('%<v0>2d %<v1>20d', v0: bqres.count, v1: bqres.first[:ky])}"
125
+ def lmult
126
+ "#{lbase} ML#{format('%<v0>2d %<v1>20d', v0: bqres.count, v1: bqres.first[:ky])}"
139
127
  end
140
128
 
141
- # obtem chaves movimento (keysin.mv) para apagar
142
129
  def add_kys
143
- bqres.each { |row| opcao[:k] += ",#{row[:ky]}" }
144
- end
145
-
146
- # @return [String] sql para movimentos no bigquery
147
- def sql_existe_mv
148
- "select * from #{BD}.gmv where nc=#{conta} and dl='#{rowfc.first.strftime(DF)}' and vl=#{rowfc[3]}"
149
- end
150
-
151
- # obtem movimento (values.mv) para inserir
152
- #
153
- # @return [String] ' NOVO'
154
- def values_mv
155
- @mvvls += ",('#{rowfc.first.strftime(DF)}','#{dvc.strftime(DF)}','#{rowfc[2]}',#{rowfc[3]},#{conta},#{dvc.year},#{dvc.month},'#{tpc}',#{ctc},null,null)"
156
- ' NOVO'
130
+ opcao[:k] << bqres.each_with_object(+'') { |r, s| s << ",#{r[:ky]}" }
157
131
  end
158
132
 
159
133
  # @return [Date] data valor corrigida
160
134
  def dvc
161
- dvl = opcao[:v]
162
- dvl.empty? ? rowfc[1] : Date.parse(dvl)
135
+ d = opcao[:v].to_s
136
+ d.empty? ? rowfc[1] : Date.parse(d)
137
+ rescue ArgumentError
138
+ rowfc[1]
163
139
  end
164
140
 
165
141
  # @return [String] classificacao do movimento (null --> classificacao automatica)
166
142
  def ctc
167
- cmv = opcao[:g]
143
+ cmv = opcao[:g].to_s
168
144
  cmv.empty? ? 'null' : "'#{cmv}'"
169
145
  end
170
146
 
data/lib/abank/version.rb CHANGED
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Abank
4
- VERSION = "0.6.8"
4
+ VERSION = '0.7.1'
5
5
  end
data/lib/abank.rb CHANGED
@@ -8,6 +8,7 @@ require('abank/version')
8
8
  # @author Hernani Rodrigues Vaz
9
9
  module Abank
10
10
  DR = "/home/#{`whoami`.chomp}/Downloads".freeze
11
+ BD = 'hernanilr.ab'
11
12
 
12
13
  # CLI para carregar folhas calculo comuns no bigquery
13
14
  class CLI < Thor
@@ -17,8 +18,9 @@ module Abank
17
18
  Big.new(options.to_h).mv_classifica.ct_dados.re_insert
18
19
  end
19
20
 
20
- desc 'apagamv', 'apaga movimentos'
21
- option :k, banner: 'KEY[,KEY...]', required: true, desc: 'keys movimentos a apagar'
21
+ desc 'apagamv', 'apaga movimentos keys|conta'
22
+ option :k, banner: 'KEY[,KEY...]', default: '', desc: 'keys movimentos apagar'
23
+ option :n, banner: 'CONTA', type: :numeric, default: 0, desc: 'conta movimentos apagar (>3 outras)'
22
24
  # apaga movimentos
23
25
  def apagamv
24
26
  Big.new(options.transform_keys(&:to_sym)).mv_delete.ct_dados.re_insert
@@ -43,13 +45,11 @@ module Abank
43
45
 
44
46
  desc 'recriact', 'atualiza rendas de contrato arrendamento'
45
47
  option :c, banner: 'CONTRATO', required: true, desc: 'Identificador contrato arrendamento'
46
- option :d, banner: 'DATA', default: '', desc: 'data contrato arrendamento'
47
- option :t, type: :boolean, default: false, desc: 'apaga todas as rendas?'
48
48
  # atualiza rendas de contrato arrendamento
49
49
  def recriact
50
50
  opc = options[:c]
51
- Big.new(c: opc, t: options[:t]).ct_apaga
52
- Big.new(c: opc, t: true, d: options[:d]).ct_cria
51
+ Big.new(c: opc, t: false).ct_apaga
52
+ Big.new(c: opc, t: true).ct_cria
53
53
  end
54
54
 
55
55
  desc 'recriare', 'atualiza rendas dos contratos ativos'
@@ -61,7 +61,8 @@ module Abank
61
61
  desc 'work', 'carrega/apaga dados da folha calculo'
62
62
  option :s, type: :boolean, default: false, desc: 'apaga movimento similar (=data,=valor,<>descricao)'
63
63
  option :e, type: :boolean, default: false, desc: 'apaga movimento igual'
64
- option :v, banner: 'DATA', default: '', desc: 'data valor para movimentos a carregar'
64
+ option :n, banner: 'CONTA', type: :numeric, default: 0, desc: 'conta destino (0 auto,1 corrente,2 cartao,3 chash,> outras)'
65
+ option :v, banner: 'DATA', default: '', desc: 'data lancamento para movimentos a carregar'
65
66
  option :g, banner: 'TAG', default: '', desc: 'classificacao para movimentos a carregar'
66
67
  # carrega/apaga dados da folha calculo
67
68
  def work
@@ -71,10 +72,11 @@ module Abank
71
72
  end
72
73
 
73
74
  desc 'show', 'mostra dados da folha calculo'
75
+ option :n, banner: 'CONTA', type: :numeric, default: 0, desc: 'conta destino (0 auto,1 corrente,2 cartao,3 chash,> outras)'
74
76
  # mostra folha calculo
75
77
  def show
76
78
  Dir.glob("#{DR}/*.xlsx").each do |file|
77
- Folha.new(options.merge(f: file)).processa_xls
79
+ Folha.new(options.transform_keys(&:to_sym).merge(f: file)).processa_xls
78
80
  end
79
81
  end
80
82
 
metadata CHANGED
@@ -1,13 +1,13 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: abank
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.6.8
4
+ version: 0.7.1
5
5
  platform: ruby
6
6
  authors:
7
7
  - Hernâni Rodrigues Vaz
8
8
  bindir: exe
9
9
  cert_chain: []
10
- date: 2025-02-16 00:00:00.000000000 Z
10
+ date: 2025-03-14 00:00:00.000000000 Z
11
11
  dependencies:
12
12
  - !ruby/object:Gem::Dependency
13
13
  name: bundler
@@ -197,7 +197,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
197
197
  - !ruby/object:Gem::Version
198
198
  version: '0'
199
199
  requirements: []
200
- rubygems_version: 3.6.3
200
+ rubygems_version: 3.6.5
201
201
  specification_version: 4
202
202
  summary: Arquiva movimentos conta-corrente, conta-cartao do activobank no bigquery.
203
203
  test_files: []