-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcsva.py
executable file
·444 lines (357 loc) · 12.1 KB
/
csva.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
#!/usr/bin/python
# -*- coding: utf-8 -*-
###############################################################################
"""
csvanywere.py
Author: Alexandre Grings
Usage: csvanywere.py [r|c|e|h] file.cyx
Where: file.cyx: query in cyx format (yaml data)
Options:
-r or -run: run query
-e or -edit: edit query
-c or -convert: convert from xqkt to cyx format
-n or --new: create empty cyx file
-p or --print: run query and print results to the screen
"""
###############################################################################
from xml.dom import minidom
import sys
import re
import os
import io
from sys import stdin
import getopt
import pyodbc
from xqkt import *
import yaml #config
from collections import OrderedDict
import argparse
import tempfile
import datetime
import traceback
import numbers
from subprocess import Popen
def exec_and_let_die(cmd_str):
cmd = cmd_str.split() #popen exige lista
proc = Popen(cmd, shell=False,
stdin=None, stdout=None, stderr=None, close_fds=True)
def str2bool(v):
return v.lower() in ("yes","true","t","1")
def isnumber(s):
try:
int(s)
return True
except ValueError:
try:
float(s)
return True
except ValueError:
return False
def isfloat(s):
try:
int(s)
return False
except ValueError:
try:
float(s)
return True
except ValueError:
return False
class CsvAnywhere():
#class variables
resumo = ""
exportar = ""
pos_exec = ""
exportar_nomes_campos = False
caracter_separacao = ";"
descricao = ""
sql_query = ""
string_conexao_windows = ""
string_conexao_pyodbc = ''
param_names = []
param_values = []
columns = [] #nomes das colunas
def extrai_parametros(self,sql_string):
regex = re.compile(r'\[([^]]*)\]',re.IGNORECASE)
params=regex.findall(sql_string)
params2=list(OrderedDict.fromkeys(params))
return params2
def substitui_parametros(self):
sqlstr=self.sql_query
if not self.param_values:
self.param_values=[]
for param in self.param_names:
print "Entre com %s:" %(param)
self.param_values+=[stdin.readline().rstrip("\n")]
for param,valor in zip(self.param_names,self.param_values):
print param+"="+valor
sqlstr = sqlstr.replace('['+param+']', valor)
return sqlstr
def config(self, attributes, key, default_value):
return attributes[key] if (key in attributes.keys()) else default_value
def get_config(self):
allowed_keys =[
"sql_query",
"resumo",
"exportar",
"pos_exec",
"exportar_nomes_campos",
"caracter_separacao",
"descricao",
"string_conexao_pyodbc",
"separador_decimal"]
return { key: self.__dict__[key] for key in allowed_keys }
def set_config(self, new_config):
""" new_config eh um dicionario """
for key in new_config:
self.__dict__[key]=new_config[key]
def le_config_teclado(self):
for key in self.get_config():
sys.stdout.write("Entre com %s (atual=%s):" %(key, self.__dict__[key]))
self.__dict__[key]=stdin.readline().rstrip("\n")
def read_config(self,filename):
# Read YAML file
with open(filename, 'r') as stream:
data_loaded = yaml.load(stream)
return data_loaded
def write_config(self,filename):
# Write YAML file
properties = self.get_config()
print properties
with io.open(filename, 'w', encoding='utf8') as outfile:
yaml.dump(properties, outfile, default_flow_style=False, allow_unicode=True)
def __init__(self,filename):
if not filename:
self.resumo = ''
self.exportar = ''
self.pos_exec = ''
self.exportar_nomes_campos = 'True'
self.caracter_separacao = ';'
self.descricao = ''
self.sql_query = ''
self.param_names = []
self.string_conexao_windows = ''
self.string_conexao_pyodbc=''
self.separador_decimal=','
return
just_filename, file_extension = os.path.splitext(filename)
self.filename=just_filename+".cyx"
if file_extension=='.xqkt':
older_xqkt = Xqkt(filename)
self.resumo = older_xqkt.resumo
self.exportar = older_xqkt.exportar
self.pos_exec = older_xqkt.pos_exec
self.exportar_nomes_campos = older_xqkt.exportar_nomes_campos
self.caracter_separacao = older_xqkt.caracter_separacao
self.descricao = older_xqkt.descricao
self.sql_query = older_xqkt.sql_query
self.param_names = older_xqkt.parametros
self.string_conexao_windows = older_xqkt.string_conexao_windows
self.string_conexao_pyodbc = older_xqkt.string_conexao_pyodbc
self.separador_decimal = ',' #nao existe no xqkt
self.write_config(self.filename) #converte para novo formato
else:
configuration = self.read_config(filename)
self.resumo = self.config(configuration,'resumo','')
self.exportar = self.config(configuration,'exportar','')
self.pos_exec = self.config(configuration,'pos_exec','')
self.exportar_nomes_campos = self.config(configuration,'exportar_nomes_campos','')
self.caracter_separacao = self.config(configuration,'caracter_separacao','')
self.descricao = self.config(configuration,'descricao','')
self.sql_query = (self.config(configuration,'sql_query','')).encode('utf-8')
self.param_names = self.extrai_parametros(self.sql_query)
self.string_conexao_windows = self.config(configuration,'string_conexao_windows','')
self.string_conexao_pyodbc=self.config(configuration,'string_conexao_pyodbc',self.string_conexao_pyodbc)
self.separador_decimal=self.config(configuration,'separador_decimal',',')
def reformat_type(self,stuff):
"""stuff pode ser string, float, decimal, etc """
if not stuff and not(isinstance(stuff, numbers.Number)):
stuff=""
str_stuff=("%s" %stuff).encode('utf-8')
if isfloat(str_stuff):
if self.separador_decimal==',':
return str_stuff.replace(',','#').replace('.',',').replace('#','.')
return str_stuff
else:
return(str_stuff)
def format_row_as_a_list(self, row_as_a_list):
list_out=[]
sizes=[]
for item in row_as_a_list:
item_str="%s"%(self.reformat_type(item))
list_out+=[item_str]
sizes+=[len(item_str)]
return (list_out,sizes)
def formatline(self, list):
size = len(list)
i = 0
linha=''
for item in list:
linha+=("%s"%(self.reformat_type(item))).encode('utf-8')
i+=1
if i < size:
linha+=self.caracter_separacao
return linha+"\n"
def connect_db(self):
self.connection = pyodbc.connect(self.string_conexao_pyodbc)
self.cursor = self.connection.cursor()
def remove_multiline_comments(self,sql):
sql_no_comments=''
abertos=0
for i in range(len(sql)):
if i < len(sql):
if sql[i:i+2]=='/*':
abertos+=1
if abertos and (sql[i-2:i]=='*/'):
abertos-=1
if abertos==0:
sql_no_comments+=sql[i]
return sql_no_comments
def remove_inline_comments(self,sql):
linhas=sql.split('\n')
no_comments=''
for linha in linhas:
no_comments+=(linha.split('#')[0]+'\n')
return no_comments
def remove_all_comments(self,sql):
return self.remove_inline_comments(self.remove_multiline_comments(sql))
def execute_query(self):
sql = self.substitui_parametros()
sql = self.remove_all_comments(sql)
self.cursor.execute(sql)
self.columns = [column[0] for column in self.cursor.description]
rows=[]
while 1:
row = self.cursor.fetchone()
if not row:
break
rows.append(row)
return rows
def htmlize_it(self,rows):
""""Convert the data to html table """
html="<table>" #this is a html table
html+="<tr>" #header
for column in self.columns:
html+="<th>%s</th> "%(column)
html+="</tr>\n" # header end
for row in rows:
html+="<tr>"
for field in row:
html+="<td>%s</td>" %(field)
html+="</tr>\n"
html+="</table>" #table end
return html
def tabularize_it(self, rows):
max_sizes=[]
for column in self.columns:
max_sizes+=[len(column)]
new_rows=[]
try:
for row in rows:
(str_list,new_sizes)=self.format_row_as_a_list(row)
max_sizes=[ a if a> b else b for a,b in zip(max_sizes,new_sizes) ]
new_rows+=[str_list]
except:
print row
raise
format_str=""
for field,size in zip(new_rows[0],max_sizes):
alignment = "%" if isnumber(field) else "%-"
format_str += alignment+str(size)+"s|"
str_rows = [ format_str % tuple(self.columns) ]
try:
for row in new_rows:
str_rows += [ format_str % tuple(row) ]
except:
print row
raise
return str_rows
def edit_it(cyx):
try:
editor=os.environ['CYX_EDITOR']
except KeyError:
print "A variavel de ambiente 'CYX_EDITOR' não existe"
print "Configure 'CYX_EDITOR' para apontar para o editor sql de sua preferência"
return
temp_filename=cyx.filename+'.tmp'
f_edit=open(temp_filename,'w')
f_edit.write(cyx.sql_query)
f_edit.close()
os.system(editor+' '+temp_filename)
f_edit=open(temp_filename,'r')
cyx.sql_query=f_edit.read()
cyx.write_config(cyx.filename)
f_edit.close()
def run_it(cyx, param_values, output_format, temp_export):
""" TODO exportar para txt ou html """
cyx.param_values=param_values
cyx.connect_db()
rows=cyx.execute_query()
if not rows:
print "A consulta nao retornou valores"
return
if output_format=="html":
print cyx.htmlize_it(rows)
return
if output_format=="txt":
for linha in cyx.tabularize_it(rows):
print linha
return
if temp_export:
cyx.exportar=temp_export[0]
if cyx.exportar or temp_export:
print "Exportando dados para arquivo "+ cyx.exportar
output_f=open(cyx.exportar,'w')
if cyx.exportar_nomes_campos:
output_f.write(cyx.formatline(cyx.columns))
for row in rows:
output_f.write(cyx.formatline(row))
output_f.close()
if temp_export:
open(cyx.exportar+".OK","a").close()
else:
print "Executando :{}".format(cyx.pos_exec)
exec_and_let_die(cyx.pos_exec)
def main(parser):
parser.add_argument("filename",help="arquivo do tipo CYX (consulta odbc e exportacao)")
parser.add_argument("-e","--edit",action="store_true",
help="Usa o editor indicado na variavel de ambiente 'CYX_EDITOR'"\
" para abrir o sql")
parser.add_argument("-n","--new",action="store_true",help="Cria uma query em branco")
parser.add_argument("-r","--run",action="store_true",help="Roda a query")
parser.add_argument("-c","--convert",action="store_true",help="Converte um arquivo xqkt em cyx")
parser.add_argument("-f","--format",choices=["html","txt","csv"], default="csv",help="Formato do output (html,txt) o defaut é csv")
parser.add_argument("-T","--temp",nargs=1,
help="Exporta dados para o arquivo FILENAME informado."\
"Ao final gera um arquivo FILENAME.ok."\
"Isso permite o intercambio de dados com outros programas."\
"O arquivo '.ok' indica que os dados estao prontos"
)
parser.add_argument("-p","--parameters",nargs='+',help="Fornece parametros na linha de comando (caso contrario o programa espera a entrada)")
args = parser.parse_args()
if args.new:
cyx=CsvAnywhere('')
cyx.le_config_teclado()
cyx.write_config(args.filename)
print "Criado arquivo %s" % (args.filename)
return
else:
cyx=CsvAnywhere(args.filename)
if args.edit:
edit_it(cyx)
elif args.run:
run_it(cyx,args.parameters, args.format, args.temp)
elif args.convert:
print "Converted" #Done when object is created
else:
print "Argumentos invalidos, veja --help"
if __name__=="__main__":
parser = argparse.ArgumentParser()
reload(sys)
sys.setdefaultencoding('utf8')
try:
main(parser)
except:
traceback.print_exc(file=sys.stdout)
print "Pressione qualquer tecla para continuar..."
stdin.readline().rstrip("\n")