Mercurial > pyarq-presupuestos
comparison Generic/fiebdc.py @ 1:2ac1551ad2ab version 0.0.0
add code
author | Miguel Ángel Bárcena Rodríguez <miguelangel@obraencurso.es> |
---|---|
date | Sun, 31 Oct 2010 20:07:33 +0100 |
parents | |
children | 6502bfdaa84d |
comparison
equal
deleted
inserted
replaced
0:a1703c4f2990 | 1:2ac1551ad2ab |
---|---|
1 #!/usr/bin/python | |
2 # -*- coding: utf-8 -*- | |
3 ## File fiebdc.py | |
4 ## This file is part of pyArq-Presupuestos. | |
5 ## | |
6 ## Copyright (C) 2010 Miguel Ángel Bárcena Rodríguez | |
7 ## <miguelangel@obraencurso.es> | |
8 ## | |
9 ## pyArq-Presupuestos is free software: you can redistribute it and/or modify | |
10 ## it under the terms of the GNU General Public License as published by | |
11 ## the Free Software Foundation, either version 3 of the License, or | |
12 ## (at your option) any later version. | |
13 ## | |
14 ## pyArq-Presupuestos is distributed in the hope that it will be useful, | |
15 ## but WITHOUT ANY WARRANTY; without even the implied warranty of | |
16 ## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
17 ## GNU General Public License for more details. | |
18 ## | |
19 ## You should have received a copy of the GNU General Public License | |
20 ## along with this program. If not, see <http://www.gnu.org/licenses/>. | |
21 | |
22 # specifications in http://www.fiebdc.org | |
23 | |
24 # Modules | |
25 import time | |
26 import re | |
27 import calendar | |
28 import os.path | |
29 | |
30 # pyArq-Presupuestos modules | |
31 import base | |
32 from Generic import utils | |
33 from Generic import globals | |
34 | |
35 class Read(object): | |
36 """fiebdc.Read: | |
37 | |
38 Description: | |
39 Reads and parses a fiebdc file | |
40 Constructor: | |
41 fiebdc.Read(filename=None, budget=None) | |
42 Ancestry: | |
43 +-- object | |
44 +-- Read | |
45 Atributes: | |
46 "__budget": budget ("base.Budget" object) | |
47 "__file_format": File format of the fiebdc file | |
48 "__format_list": List of file format that can be readed | |
49 "__character_sets_dict": Dictionary with the character sets supported | |
50 "__character_set": character_set of the file | |
51 "__generator": program which the file is created | |
52 "__cancel": Boolean value, True mean that the read process must stop | |
53 "__filename": The filename of the fiebdc file that is readed | |
54 "__pattern": re compiled pattern dict | |
55 Methods: | |
56 __init__(self, filename=None, budget=None) | |
57 cancel(self) | |
58 eraseControlCharacters(self, string) | |
59 validateCode(self, code) | |
60 parseDate(self, date) | |
61 parseRecord(self,record) | |
62 _parseV(self, field_list) | |
63 _parseC(self, field_list) | |
64 _parseDY(self, field_list) | |
65 _parseMN(self, field_list) | |
66 _parseT(self, field_list) | |
67 _parseK(self, field_list) | |
68 _parseW(self, field_list) | |
69 _parseL(self, field_list) | |
70 _parseQ(self, field_list) | |
71 _parseJ(self, field_list) | |
72 _parseG(self, field_list) | |
73 _parseE(self, field_list) | |
74 _parseX(self, field_list) | |
75 _parseF(self, field_list) | |
76 readFile(self, budget=None, filename=None) | |
77 """ | |
78 def __init__(self, filename=None, budget=None): | |
79 """def __init__(self, filename=None, budget=None) | |
80 | |
81 Sets the instance attributes | |
82 """ | |
83 self.__budget = budget | |
84 self.__filename = filename | |
85 if not self.__budget is None: | |
86 self.__budget.filename = self.__filename | |
87 self.__cancel = False | |
88 self.__format_list = ["FIEBDC-3/95", "FIEBDC-3/98", "FIEBDC-3/2002", | |
89 "FIEBDC-3/2004", "FIEBDC-3/2007"] | |
90 # ANSI->¿"ISO-8859-15" or "latin1 ISO-8859-1" or "cp1252 windows-1252"? | |
91 # 850 -> IBM850 -> cp850 | |
92 # 437 -> IBM437 -> cp437 | |
93 self.__character_sets_dict = {"ANSI" : "cp1252", | |
94 "850" : "850", | |
95 "437" : "cp437"} | |
96 self.__file_format = "FIEBDC-3/2007" | |
97 self.__generator = globals.version | |
98 self.__character_set = "850" | |
99 self.__pattern = { | |
100 "control_tilde" : re.compile("((\r\n)| |\t)+~"), | |
101 "control_vbar" : re.compile("((\r\n)| |\t)+\|"), | |
102 "control_backslash" : re.compile(r"((\r\n)| |\t)+\\"), | |
103 "valid_code" : re.compile("[^A-Za-z0-9ñÑ.$#%&_]"), | |
104 "special_char": re.compile("[#%&]"), | |
105 "no_float": re.compile("[^0-9.]"), | |
106 "formula" : re.compile(".*[^0123456789\.()\+\-\*/\^abcdp ].*"), | |
107 "comment": re.compile("#.*\r\n"), | |
108 "empty_line": re.compile(r"(\r\n) *\r\n"), | |
109 "space_before_backslash" : re.compile(r"( )+\\"), | |
110 "space_after_backslash" : re.compile(r"\\( )+"), | |
111 "start_noend_backslash" : re.compile("(\r\n\\\.*[^\\\])\r\n"), | |
112 "end_oper": re.compile("(\+|-|\*|/|/^|@|&|<|>|<=|>=|=|!) *\r\n"), | |
113 "matricial_var" : re.compile("(\r\n *[%|\$][A-ZÑ].*=.*,) *\r\n"), | |
114 "descomposition" : re.compile("^([^:]+):(.*)$"), | |
115 "var" : re.compile("^([$%][A-ZÑ][()0-9, ]*)=(.*)$"), | |
116 "after_first_tilde" : re.compile("^[^~]*~"), | |
117 "end_control" : re.compile("((\r\n)| |\t)+$"), | |
118 } | |
119 | |
120 def cancel(self): | |
121 """def cancel(self) | |
122 | |
123 Sets the "__cancel" attribute to True, It stops the read process. | |
124 """ | |
125 self.__cancel = True | |
126 | |
127 def eraseControlCharacters(self, string): | |
128 """eraseControlCharacters(self,string) | |
129 | |
130 Return a copy of the string with the blank characters (32), | |
131 tabs (9) and end of line (13 and 10) before of the separators | |
132 '~', '|' erased. | |
133 Before separator \ not deleted because it affects the reading of the | |
134 record ~P | |
135 """ | |
136 # "control_tilde" : "((\r\n)| |\t)+~" | |
137 string = self.__pattern["control_tilde"].sub("~",string) | |
138 # "control_vbar" : "((\r\n)| |\t)+\|" | |
139 string = self.__pattern["control_vbar"].sub("|",string) | |
140 # "control_backslash" : r"((\r\n)| |\t)+\\" | |
141 #string = self.__pattern["control_backslash"].sub(r"\\",string) | |
142 return string | |
143 | |
144 def validateCode(self, code): | |
145 """validateCode(self, code) | |
146 | |
147 Test if the code have invalid characters and try to erase it, | |
148 if it is posible return a valid code else return a empty string. | |
149 """ | |
150 if not isinstance(code, str): | |
151 print _("Invalid code, it must be a string") | |
152 return "" | |
153 # Valid chararcter: A-Z a-z 0-9 ñ Ñ . $ # % & _ | |
154 # "valid_code" : "[^A-Za-z0-9ñÑ.$#%&_]" | |
155 _code = self.__pattern["valid_code"].sub("", code) | |
156 if _code != code: | |
157 print utils.mapping(_("The code '$1' have invalid characters."), | |
158 (code,)) | |
159 code = _code | |
160 # the lasts characters can not be <#> or <##> | |
161 # <##> -> root record in FIEFDC-3 | |
162 # <#> -> chapter record in FIEFDC-3 | |
163 if len(code) > 0: | |
164 while code[-1] == "#": | |
165 code = code[:-1] | |
166 if len(code) > 20: | |
167 code = code[:20] | |
168 # only one charecter # % or & | |
169 if sum([code.count(c) for c in '#%&']) > 1: | |
170 print utils.mapping(_("The code '$1' contains special "\ | |
171 "characters repeated."),(code,)) | |
172 _i = min([code.find(c) for c in '#%&']) | |
173 code = code[:_i+1] + \ | |
174 self.__pattern["special_char"].sub("", code[_i+1:]) | |
175 return code | |
176 | |
177 def parseDate(self, date): | |
178 """parseDate(self, date) | |
179 | |
180 date: in the format: | |
181 uneven len: add a Leading 0 | |
182 len = 8 DDMMYYYY | |
183 len <= 6 DDMMYY “80/20”. >80 -> >1980 <80 -> <2080 | |
184 len < 5 MMYY | |
185 len < 3 YY | |
186 Test date string and return a tuple (YYYY, MM, DD) | |
187 or None if the date format is invalid | |
188 """ | |
189 # All characters must be numbers, len <= 8 and not empty string | |
190 if not date.isdigit() or len(date) > 8 or date == "": | |
191 return None | |
192 else: | |
193 if len(date)%2 == 1: # uneven len: add a leading 0 | |
194 date = "0" + date | |
195 if len(date) == 8: | |
196 _d = int(date[:2]) | |
197 _m = int(date[2:4]) | |
198 _y = int(date[4:8]) | |
199 elif len(date) <= 6: | |
200 _y = int(date[-2:]) | |
201 if _y < 80: _y = 2000 + _y | |
202 else: _y = 1900 + _y | |
203 if len(date) == 6: | |
204 _d = int(date[:2]) | |
205 _m = int(date[2:4]) | |
206 elif len(date) == 4: | |
207 _d = 0 | |
208 _m = int(date[:2]) | |
209 elif len(date) == 2: | |
210 _d = 0 | |
211 _m = 0 | |
212 if not _d in range(1,31): _d = 0 | |
213 if not _m in range(1,12): _m = 0 | |
214 if _m == 0: _d = 0 | |
215 if _m != 0 and _d != 0: | |
216 if calendar.monthrange(_y, _m)[1] < _d: | |
217 _d = 0 | |
218 return (_y, _m, _d) | |
219 | |
220 def parseRecord(self,record): | |
221 """parseRecord(self,record) | |
222 | |
223 record: the record line readed from the file whith the format: | |
224 type|field|field|subfield\subfield|... | |
225 [a] nothing or "a" | |
226 {a} zero or more #-#twice#-# "a" | |
227 <a> one or more #-#twice#-# "a" | |
228 Types: V C D Y M N T K L Q J G E X B F A | |
229 V: Property and Version | |
230 1- [File_Owner] | |
231 2- Format_Version[\DDMMYYYY] | |
232 3- [Program_Generator] | |
233 4- [Header]\{Title\} | |
234 5- [Chaters_set] | |
235 6- [Comment] | |
236 C: Record: | |
237 1- Code{\Code} | |
238 2- [Unit] | |
239 3- [Summary] | |
240 4- {Price\} | |
241 5- {Date\} | |
242 6- [Type] | |
243 D or Y: DECOMPOSITION or ADD DECOMPOSITION | |
244 1- Parent Code | |
245 2- <Child Code\ [Factor]\ [Yield]> | |
246 M or N: MEASURE or ADD MEASURE | |
247 1- [Parent Code\]Child Code | |
248 2- {Path\} | |
249 3- TOTAL MEASURE | |
250 4- {Type\Comment\Unit\Length\Width\Height\} | |
251 5- [Label] | |
252 T: Text | |
253 1- Code | |
254 2- Description text | |
255 K: Coefficients | |
256 1- { DN \ DD \ DS \ DR \ DI \ DP \ DC \ DM \ DIVISA \ } | |
257 2- CI \ GG \ BI \ BAJA \ IVA | |
258 3- { DRC \ DC \ DRO \ DFS \ DRS \ DFO \ DUO \ DI \ DES \ DN \ | |
259 DD \ DS \ DIVISA \ } | |
260 4- [ n ] | |
261 L: Sheet of Conditions 1 | |
262 A) | |
263 1- Empty | |
264 2- {Section Code\Section Title} | |
265 B) | |
266 1- Record Code | |
267 2- {Section Code\Section Text} | |
268 3- {Section Code\RTF file} | |
269 4- {Section Code\HTM file} | |
270 Q: Sheet of Conditions 2 | |
271 1- Record Code | |
272 2- {Section Code\Paragraph key\{Field key;}\}| | |
273 J: Sheet of Conditions 3 | |
274 1- Paragraph code | |
275 2- [Paragraph text] | |
276 3- [RTF file] | |
277 4- [HTML file] | |
278 G: Grafic info | |
279 1- <grafic_file.ext\> | |
280 E: Company | |
281 1- company Code | |
282 2 [ summary ] | |
283 3- [ name ] | |
284 4- { [ type ] \ [ subname ] \ [ address ] \ [ postal_code ] | |
285 \ [ town ] \ [ province ] \ [ country ] \ { phone; } | |
286 \ { fax; } \ {contact_person; } \ } | |
287 5- [ cif ] \ [ web ] \ [ email ] \ | |
288 X: Tecnical information | |
289 A) | |
290 1- Empty | |
291 2- < TI_Code \ TI_Descitption \ TI_Unit > | |
292 B) | |
293 1- Record_code | |
294 2- < TI_Code \ TI_value > | |
295 F: #-#Adjunto#-# File | |
296 1- Record code | |
297 2- { Type \ { Filenames; } \ [Description] } | |
298 B: Change code | |
299 1- Record Code | |
300 2- New code | |
301 A: Labels | |
302 1- Record Code | |
303 2- <Label\> | |
304 """ | |
305 # TODO: ~L ~J RTF and HTML files | |
306 # TODO: test ~Q ~J ~G | |
307 # TODO: ~P. Registro tipo Descripción Paramétrica. | |
308 # TODO: ~O. Registro tipo Relación Comercial. | |
309 # TODO: test records | |
310 _field_list = record.split("|") | |
311 self._record_number = self._record_number +1 | |
312 _budget = self.__budget | |
313 if _field_list[0] == "V": | |
314 self._record_V_number += 1 | |
315 self._parseV(_field_list) | |
316 elif _field_list[0] == "C": | |
317 self._record_C_number += 1 | |
318 self._parseC(_field_list) | |
319 elif _field_list[0] == "D": | |
320 self._record_D_number += 1 | |
321 self._parseDY(_field_list) | |
322 elif _field_list[0] == "Y": | |
323 self._record_Y_number += 1 | |
324 self._parseDY(_field_list) | |
325 elif _field_list[0] == "M": | |
326 self._record_M_number += 1 | |
327 self._parseMN(_field_list) | |
328 elif _field_list[0] == "N": | |
329 self._record_N_number += 1 | |
330 self._parseMN(_field_list) | |
331 elif _field_list[0] == "T": | |
332 self._record_T_number += 1 | |
333 self._parseT(_field_list) | |
334 elif _field_list[0] == "K": | |
335 self._record_K_number += 1 | |
336 self._parseK(_field_list) | |
337 elif _field_list[0] == "W": | |
338 self._record_W_number += 1 | |
339 self._parseW(_field_list) | |
340 elif _field_list[0] == "L": | |
341 self._record_L_number += 1 | |
342 self._parseL(_field_list) | |
343 elif _field_list[0] == "Q": | |
344 self._record_Q_number += 1 | |
345 self._parseQ(_field_list) | |
346 elif _field_list[0] == "J": | |
347 self._record_J_number += 1 | |
348 self._parseJ(_field_list) | |
349 elif _field_list[0] == "G": | |
350 self._record_G_number += 1 | |
351 self._parseG(_field_list) | |
352 elif _field_list[0] == "E": | |
353 self._record_E_number += 1 | |
354 self._parseE(_field_list) | |
355 elif _field_list[0] == "O": | |
356 self._record_O_number += 1 | |
357 elif _field_list[0] == "P": | |
358 self._record_P_number += 1 | |
359 self._parseP(_field_list) | |
360 elif _field_list[0] == "X": | |
361 self._record_X_number += 1 | |
362 self._parseX(_field_list) | |
363 elif _field_list[0] == "B": | |
364 self._record_B_number += 1 | |
365 self._parseB(_field_list) | |
366 elif _field_list[0] == "F": | |
367 self._record_F_number += 1 | |
368 self._parseF(_field_list) | |
369 elif _field_list[0] == "A": | |
370 self._record_A_number += 1 | |
371 self._parseA(_field_list) | |
372 else: | |
373 self._record_Unknow_number += 1 | |
374 | |
375 def _parseV(self, field_list): | |
376 """_parseV(self, field_list) | |
377 | |
378 field_list: field list of the record | |
379 0- V :Property and Version | |
380 1- [File_Owner] | |
381 2- Format_Version[\DDMMYYYY] | |
382 3- [Program_Generator] | |
383 4- [Header]\{Title\} | |
384 5- [Chaters_set] | |
385 6- [Comment] | |
386 7- [Data type] | |
387 8- [Number budget certificate] | |
388 9- [Date budget certificate] | |
389 """ | |
390 if self._record_number != 1: | |
391 print utils.mapping(_("The 'V' record (Property and Version) "\ | |
392 "must be the first record in the file but it is the "\ | |
393 "number: $1"), (self._record_number,)) | |
394 print _("The default values were taken and this V record is "\ | |
395 "ignored") | |
396 return | |
397 # _____number of fields_____ | |
398 # Any INFORMATION after last field separator is ignored | |
399 if len(field_list) > 10: | |
400 field_list = field_list[:10] | |
401 # If there are no sufficient fields, the fields are added | |
402 # with empty value:"" | |
403 else: | |
404 field_list = field_list + [""]*(10-len(field_list)) | |
405 # control character are erased: end of line, tab, space | |
406 # only leading and trailing whitespace in owner, generator, comment | |
407 # _____Fields_____ | |
408 _record_type = self.delete_control_space(field_list[0]) | |
409 _owner = field_list[1].strip() | |
410 _owner = self.delete_control(_owner) | |
411 _version_date = self.delete_control_space(field_list[2]) | |
412 _generator = field_list[3].strip() | |
413 _generator = self.delete_control(_generator) | |
414 _header_title = field_list[4].strip() | |
415 _header_title = self.delete_control(_header_title) | |
416 _character_set = self.delete_control_space(field_list[5]) | |
417 _comment = field_list[6].strip("\t \n\r") | |
418 _data_type = self.delete_control_space(field_list[7]) | |
419 _number_certificate = self.delete_control_space(field_list[8]) | |
420 __date_certificate = self.delete_control_space(field_list[9]) | |
421 # _____Owner_____ | |
422 self.__budget.setOwner(_owner) | |
423 # _____Version-Date_____ | |
424 _version_date = _version_date.split("\\") | |
425 _file_format = _version_date[0] | |
426 if _file_format in self.__format_list: | |
427 self.__file_format = _file_format | |
428 print _("FIEBDC format: %s" % _file_format) | |
429 if len(_version_date) > 1: | |
430 _date = _version_date[1] | |
431 if _date != "": | |
432 _parsed_date = self.parseDate(_date) | |
433 if _parsed_date is not None: | |
434 self.__budget.setDate(_parsed_date) | |
435 # _____Generator_____ | |
436 # ignored field | |
437 print _("FIEBDC file generated by %s" % _generator) | |
438 # _____Header_Title_____ | |
439 _header_title = _header_title.split("\\") | |
440 _header_title = [_title.strip() for _title in _header_title] | |
441 _header = _header_title.pop(0) | |
442 _title = [ ] | |
443 for _title_index in _header_title: | |
444 if _title_index != "": | |
445 _title.append(_title_index) | |
446 if _header != "": | |
447 self.__budget.setTitleList([ _header, _title ]) | |
448 # _____Characters_set_____ | |
449 # field parsed in readFile method | |
450 # _____Comment_____ | |
451 if _comment != "": | |
452 self.__budget.setComment(_comment) | |
453 # _____Data type_____ | |
454 # 1 -> Base data. | |
455 # 2 -> Budget. | |
456 # 3 -> Budget certificate. | |
457 # 4 -> Base date update. | |
458 try: | |
459 _data_type = int(_data_type) | |
460 except ValueError: | |
461 _data_type = "" | |
462 if _data_type == 3: | |
463 # _____Number budget certificate_____ | |
464 try: | |
465 _number_certificate = int(_number_certificate) | |
466 except ValueError: | |
467 _number_certificate = "" | |
468 # _____Date budget certificate_____ | |
469 if _date_certificate != "": | |
470 _parsed_date_certificate = self.parseDate(_date_certificate) | |
471 if _parsed_date_certificate is None: | |
472 _date_certificate = "" | |
473 else: | |
474 _date_certificate = _parsed_date_certificate | |
475 self.__budget.setBudgetype(_data_type) | |
476 self.__budget.setCertificateOrder(_number_certificate) | |
477 self.__budget.setCertificateDate(_parsed_date_cerfificate) | |
478 elif _data_type != "": | |
479 self.__budget.setBudgeType(_data_type) | |
480 self.num_valid_record = self.num_valid_record + 1 | |
481 | |
482 def _parseK(self, field_list): | |
483 """_parseK(self, field_list) | |
484 | |
485 field_list: field list of the record | |
486 0- K: Coefficients | |
487 1- { DN \ DD \ DS \ DR \ DI \ DP \ DC \ DM \ DIVISA \ } | |
488 2- CI \ GG \ BI \ BAJA \ IVA | |
489 3- | |
490 A){ DRC \ DC \ DRO \ DFS \ DRS \ DFO \ DUO \ DI \ DES \ DN \ | |
491 DD \ DS \ DIVISA \ } | |
492 B){ DRC \ DC \ \ DFS \ DRS \ \ DUO \ DI \ DES \ DN \ | |
493 DD \ DS \ DSP\ DEC\ DIVISA \ } | |
494 4- [ n ] | |
495 """ | |
496 # _____Number of fields_____ | |
497 # Any INFORMATION after last field separator is ignored | |
498 # The record must have 3 fields | |
499 # The last field is ignored, pyArq hate dll's | |
500 if len(field_list) > 4: | |
501 field_list = field_list[1:4] | |
502 # The record must have 3 fields | |
503 else: | |
504 field_list = field_list[1:] + [""]*(4-len(field_list)) | |
505 # control character are erased: end of line, tab, space | |
506 # _____Fields_____ | |
507 _field0 = self.delete_control_space(field_list[0]) | |
508 _field1 = self.delete_control_space(field_list[1]) | |
509 _field2 = self.delete_control_space(field_list[2]) | |
510 # _____Field 1_____ | |
511 if len(_field1) > 0 and _field1[-1] == "\\": | |
512 _field1 = _field1[:-1] | |
513 # if there are a \ character at the end it must be erased | |
514 _percentages = _field1.split("\\") | |
515 if len(_percentages) > 5: | |
516 _percentages = _percentages[:5] | |
517 # If there are no sufficient subfields, the subfields are added | |
518 # with empty value:"" | |
519 else: | |
520 _percentages = _percentages + [""]*(5-len(_percentages)) | |
521 _percentage_titles = [ "CI", "GG", "BI", "BAJA", "IVA" ] | |
522 _percentage_dict = {} | |
523 for _percentage_index in range(len(_percentages)): | |
524 try: | |
525 _percentage = int(_percentages[_percentage_index]) | |
526 except ValueError: | |
527 _percentage = "" | |
528 _percentage_dict[_percentage_titles[_percentage_index]] = \ | |
529 _percentage | |
530 self.__budget.setPercentages(_percentage_dict) | |
531 # _____Field 0 and 1_____ | |
532 # Default number of decimal places | |
533 # Number of titles in ~V record | |
534 _title_num = len(self.__budget.getTitleList()[1]) | |
535 if _title_num == 0: _title_num = 1 | |
536 # If the field 2 is empty, the field 0 is readed | |
537 if _field2 == "": | |
538 # _____Field 0_____ | |
539 if _field0[-1] == "\\": | |
540 _field0 = _field0[:-1] | |
541 # if there are a \ character at the end it must be erased | |
542 _decimal_list = _field0.split("\\") | |
543 _decimal_index = 0 | |
544 if len(_decimal_list)%9 != 0: | |
545 # if it is not multiple of 9, empty subfield are added | |
546 _decimal_list = _decimal_list + [""]*(9 - \ | |
547 len(_decimal_list)%9) | |
548 # The number of decimal values is the same as the numbers of | |
549 # titles in the V record | |
550 if len(_decimal_list)//9 > _title_num: | |
551 _decimal_list = _decimal_list[:_title_num*9] | |
552 elif len(_decimal_list)//9 < _title_num: | |
553 _decimal_list = _decimal_list + _decimal_list[-9:] * \ | |
554 (_title_num-(len(_decimal_list)//9)) | |
555 while _decimal_index <= len(_decimal_list)-9: | |
556 _decimals = _decimal_list[_decimal_index:(_decimal_index + 9)] | |
557 _forlist = range(len(_decimals)-1) | |
558 for _index in range(len(_decimals)): | |
559 try: | |
560 #TODO: test this | |
561 _decimals[_index] = int(_decimals[_index]) | |
562 except ValueError: | |
563 _decimals[_index] = "" | |
564 _DN = _decimals[0] | |
565 _DD = _decimals[1] | |
566 _DS = _decimals[2] | |
567 _DR = _decimals[3] | |
568 _DI = _decimals[4] | |
569 _DP = _decimals[5] | |
570 _DC = _decimals[6] | |
571 _DM = _decimals[7] | |
572 _DIVISA = _decimals[8] | |
573 _percentage_dict = {"DN" : _DN, | |
574 "DD" : _DD, | |
575 "DSP" : _DS, | |
576 "DS" : _DS, | |
577 "DFC" : _DR, | |
578 "DFPU" : _DR, | |
579 "DFUO" : _DR, | |
580 "DFA" : _DR, | |
581 "DRC" : _DR, | |
582 "DRPU" : _DR, | |
583 "DRUO" : _DR, | |
584 "DRA" : _DR, | |
585 "DP" : _DC, | |
586 "DC" : _DC, | |
587 "DPU" : _DC, | |
588 "DUO" : _DC, | |
589 "DEA" : _DC, | |
590 "DES" : _DC, | |
591 "DIR" : _DI, | |
592 "DIRC" : _DI, | |
593 "DCD" : _DP, | |
594 "DIVISA": _DIVISA } | |
595 _decimal_index = _decimal_index + 9 | |
596 self.__budget.setDecimals(_percentage_dict, | |
597 (_decimal_index//9)) | |
598 else: | |
599 # _____Field 3_____ | |
600 if _field2[-1] == "\\": | |
601 _field2 = _field2[:-1] | |
602 # if there are a \ character at the end it must be erased | |
603 _decimal_list = _field2.split("\\") | |
604 # test if the Divisa subfield is 12 or 14 position | |
605 # Divisa is the only Alphanumeric subfield | |
606 # "no_float": "[^0-9.]" | |
607 if len(_decimal_list) >= 13 and \ | |
608 self.__pattern["no_float"].search(_decimal_list[12]): | |
609 _multiple = 13 | |
610 elif len(_decimal_list) >= 15 and \ | |
611 self.__pattern["no_float"].search(_decimal_list[14]): | |
612 _multiple = 15 | |
613 else: | |
614 if self.__file_format == "FIEBDC-3/2002": | |
615 _multiple = 13 | |
616 elif self.__file_format == "FIEBDC-3/2004": | |
617 _multiple = 13 | |
618 elif self.__file_format == "FIEBDC-3/2007": | |
619 _multiple = 15 | |
620 else: | |
621 _multiple = 15 | |
622 _decimal_index = 0 | |
623 if len(_decimal_list)%_multiple != 0 : | |
624 # if it is not multiple of _multiple, empty subfield are added | |
625 _decimal_list = _decimal_list + \ | |
626 [""]*(_multiple-len(_decimal_list)%_multiple) | |
627 # The number of decimal values is the same as the numbers of | |
628 # titles in the V record | |
629 if len(_decimal_list)//_multiple > _title_num: | |
630 _decimal_list = _decimal_list[:_title_num*_multiple] | |
631 elif len(_decimal_list)//_multiple < _title_num: | |
632 _decimal_list = _decimal_list + [_decimal_list[-_multiple:]]*\ | |
633 (_title_num-(len(_decimal_list)//_multiple)) | |
634 while _decimal_index <= len(_decimal_list)-_multiple: | |
635 _decimals = _decimal_list[_decimal_index:(_decimal_index +\ | |
636 _multiple)] | |
637 for _index in range(len(_decimals)-1): | |
638 try: | |
639 _decimals[_index] = int(_decimals[_index]) | |
640 except: | |
641 _decimals[_index] = "" | |
642 if _multiple == 13: | |
643 _DRC = _decimals[0] | |
644 _DC = _decimals[1] | |
645 _DRO = _decimals[2] | |
646 _DFS = _decimals[3] | |
647 _DRS = _decimals[4] | |
648 _DFO = _decimals[5] | |
649 _DUO = _decimals[6] | |
650 _DI = _decimals[7] | |
651 _DES = _decimals[8] | |
652 _DN = _decimals[9] | |
653 _DD = _decimals[10] | |
654 _DS = _decimals[11] | |
655 _DIVISA = _decimals[12] | |
656 _percentage_dict = { | |
657 "DN" : _DN, | |
658 "DD" : _DD, | |
659 "DSP" : _DS, | |
660 "DS" : _DS, | |
661 "DFC" : _DFS, | |
662 "DFPU" : _DRC, | |
663 "DFUO" : _DFS, | |
664 "DFA" : _DFS, | |
665 "DRC" : _DRS, | |
666 "DRPU" : _DRC, | |
667 "DRUO" : _DRS, | |
668 "DRA" : _DRS, | |
669 "DP" : _DC, | |
670 "DC" : _DC, | |
671 "DPU" : _DC, | |
672 "DUO" : _DUO, | |
673 "DEA" : _DES, | |
674 "DES" : _DES, | |
675 "DIR" : _DI, | |
676 "DIRC" : _DC, | |
677 "DCD" : _DI, | |
678 "DIVISA": _DIVISA, | |
679 } | |
680 else: # _multiple == 15: | |
681 _DRC = _decimals[0] | |
682 _DC = _decimals[1] | |
683 _DRO = _decimals[2] | |
684 _DFS = _decimals[3] | |
685 _DRS = _decimals[4] | |
686 _DFO = _decimals[5] | |
687 _DUO = _decimals[6] | |
688 _DI = _decimals[7] | |
689 _DES = _decimals[8] | |
690 _DN = _decimals[9] | |
691 _DD = _decimals[10] | |
692 _DS = _decimals[11] | |
693 _DSP = _decimals[12] | |
694 _DEC = _decimals[13] | |
695 _DIVISA = _decimals[14] | |
696 _percentage_dict = { | |
697 "DN" : _DN, | |
698 "DD" : _DD, | |
699 "DSP" : _DSP, | |
700 "DS" : _DS, | |
701 "DFC" : _DFS, | |
702 "DFPU" : _DRC, | |
703 "DFUO" : _DFS, | |
704 "DFA" : _DFS, | |
705 "DRC" : _DRS, | |
706 "DRPU" : _DRC, | |
707 "DRUO" : _DRS, | |
708 "DRA" : _DRS, | |
709 "DP" : _DC, | |
710 "DC" : _DC, | |
711 "DPU" : _DC, | |
712 "DUO" : _DUO, | |
713 "DEA" : _DEC, | |
714 "DES" : _DES, | |
715 "DIR" : _DI, | |
716 "DIRC" : _DC, | |
717 "DCD" : _DI, | |
718 "DIVISA": _DIVISA} | |
719 _decimal_index = _decimal_index + 13 | |
720 self.__budget.setDecimals(_percentage_dict, | |
721 (_decimal_index//13)) | |
722 self.num_valid_record = self.num_valid_record +1 | |
723 | |
724 def _parseC(self, field_list): | |
725 """_parseC(self, field_list) | |
726 | |
727 field_list: field list of the record | |
728 0- C: Record | |
729 1- Code{\Code} | |
730 2- [Unit] | |
731 3- [Summary] | |
732 4- {Price\} | |
733 5- {Date\} | |
734 6- [Type] | |
735 """ | |
736 # _____number of fields_____ | |
737 # Any INFORMATION after last field separator is ignored | |
738 if len(field_list) > 7: | |
739 field_list = field_list[:7] | |
740 # If there are no sufficient fields, the fields are added | |
741 # with empty value:"" | |
742 else: | |
743 field_list = field_list + [""]*(7-len(field_list)) | |
744 # control character are erased: en of line, tab, space | |
745 # _____Fields_____ | |
746 _record_type = field_list[0] | |
747 _codes = self.delete_control_space(field_list[1]) | |
748 _unit = self.delete_control_space(field_list[2]) | |
749 _summary = self.delete_control(field_list[3]) | |
750 _prices = self.delete_control_space(field_list[4]) | |
751 _dates = self.delete_control_space(field_list[5]) | |
752 _type = self.delete_control_space(field_list[6]) | |
753 # _____Code_____ | |
754 _codes = _codes.split("\\") | |
755 if len(_codes) > 0: | |
756 # parse the hierarchy of the first code | |
757 # hierarchy: 0->root, 1->Chapter/subchapter, 2->other | |
758 if len(_codes[0]) > 2 and _codes[0][-2:] == "##": | |
759 _hierarchy = 0 | |
760 elif len(_codes[0]) > 1 and _codes[0][-1:] == "#": | |
761 _hierarchy = 1 | |
762 else: | |
763 _hierarchy = 2 | |
764 # "#" and "##" characters at the end of the code are erased | |
765 # invalid characters are also erased | |
766 # maximun len 20 characters | |
767 _codes = [self.validateCode(_code) for _code in _codes] | |
768 # empty codes are ignored | |
769 while "" in _codes: | |
770 _codes.remove("") | |
771 if len(_codes) > 0: | |
772 #TODO: test this | |
773 _code = _codes[0] | |
774 _synonyms = _codes | |
775 else: | |
776 print _("Record C without a valid code") | |
777 return | |
778 # _____Unit_____ | |
779 # nothing to do | |
780 # _____Summary_____ | |
781 # nothing to do | |
782 # _____Price_____ and _____Dates_____ | |
783 # last \ is erased | |
784 if len(_dates) > 0 and _dates[-1] == "\\": | |
785 _dates = _dates[:-1] | |
786 if len(_prices) > 0 and _prices[-1] == "\\": | |
787 _prices = _prices[:-1] | |
788 _dates = _dates.split("\\") | |
789 _prices = _prices.split("\\") | |
790 # number of prices = number of titles in "V" line | |
791 # if there are no sufficient prices it takes the last price defined | |
792 _title_num = len(self.__budget.getTitleList()[1]) | |
793 if _title_num == 0: _title_num = 1 | |
794 if len(_prices) > _title_num: _prices = _prices[:_title_num] | |
795 elif len(_prices) < _title_num: | |
796 _prices = _prices + [_prices[-1]]*(_title_num-len(_prices)) | |
797 # number of dates = number of prices | |
798 # if there are no sufficient dates it takes the last date defined | |
799 if len(_dates) > len(_prices): _dates = _dates[:len(_prices)] | |
800 elif len(_dates) < len(_prices): | |
801 _dates = _dates + [_dates[-1]]*(len(_prices)-len(_dates)) | |
802 for _index in range(len(_prices)): | |
803 # TODO: lack to specify the number of decimals of the price | |
804 try: | |
805 _prices[_index] = float(_prices[_index]) | |
806 except: | |
807 _prices[_index] = 0.0 | |
808 _parsed_date = self.parseDate(_dates[_index]) | |
809 if _parsed_date is None: | |
810 _dates[_index] = "" | |
811 else: | |
812 _dates[_index] = _parsed_date | |
813 # _____Type_____ | |
814 # 0 Without classifying | |
815 # EA Auxiliary element | |
816 # EU Unitary element | |
817 # EC Complex element | |
818 # EF Functional element | |
819 # OB Construction site | |
820 # PA Cost overrun | |
821 # PU Unitary budget | |
822 # 1 Labourforce | |
823 # H Labourforce | |
824 # 2 Machinery and auxiliary equipment | |
825 # Q Machinery | |
826 # % Auxiliary equipment | |
827 # 3 Building materials | |
828 # MC Cement | |
829 # MCr Ceramic | |
830 # MM Wood | |
831 # MS Iron and steel | |
832 # ME Energy | |
833 # MCu Copper | |
834 # MAl Aluminium | |
835 # ML Bonding agents | |
836 # M Others materials | |
837 # Hierarchy type subtype | |
838 # 0->root -> 0 -> None,OB | |
839 # 1->[sub]chapter -> 0 -> None,PU | |
840 # 2->Other -> 0 -> None,EA,EU,EC,EF,PA | |
841 # 1 -> None,H | |
842 # 2 -> None,Q,% | |
843 # 3 -> None,MC,MCr,MM,MS,ME,MCu,Mal,ML,M | |
844 if _hierarchy == 0: | |
845 if _type == "OB": | |
846 _subtype = _type | |
847 _type = 0 | |
848 elif _type == "0" or _type == "": | |
849 _subtype = "" | |
850 _type = 0 | |
851 else: | |
852 print utils.mapping(_("Incorrect type ($1) in the code $2"), | |
853 (str(_type), _code)) | |
854 _type = 0 | |
855 _subtype = "" | |
856 elif _hierarchy == 1: | |
857 if _type == "PU": | |
858 _subtype = _type | |
859 _type = 0 | |
860 elif _type == "0" or _type == "": | |
861 _subtype = "" | |
862 _type = 0 | |
863 else: | |
864 print utils.mapping(_("Incorrect type ($1) in the code $2"), | |
865 (str(_type), _code)) | |
866 _type = 0 | |
867 _subtype = "" | |
868 else: | |
869 if _type == "EA" or _type == "EU" or _type == "EC" or \ | |
870 _type == "EF" or _type == "PA": | |
871 _subtype = _type | |
872 _type = 0 | |
873 elif _type == "H": | |
874 _subtype = _type | |
875 _type = 1 | |
876 elif _type == "Q" or _type == "%": | |
877 _subtype = _type | |
878 _type = 2 | |
879 elif _type == "MC" or _type == "MCr" or _type == "MM" or \ | |
880 _type == "MS" or _type == "ME" or _type == "MCu" or \ | |
881 _type == "Mal" or _type == "ML" or _type == "M": | |
882 _subtype = _type | |
883 _type = 3 | |
884 elif _type == "0" or _type == "1" or _type == "2" or \ | |
885 _type == "3": | |
886 _subtype = "" | |
887 _type = int(_type) | |
888 elif _type == "": | |
889 _subtype = "" | |
890 _type = 0 | |
891 else: | |
892 print utils.mapping(_("Incorrect type ($1) in the code $2"), | |
893 (str(_type), _code)) | |
894 _type = 0 | |
895 _subtype = "" | |
896 self.__budget.setRecord(_code, _synonyms, _hierarchy, | |
897 _unit, _summary, _prices, _dates, _type, _subtype) | |
898 self.num_valid_record = self.num_valid_record + 1 | |
899 | |
900 def _parseDY(self, field_list): | |
901 """_parseDY(self, field_list) | |
902 | |
903 field_list: field list of the record | |
904 0- D or Y: DECOMPOSITION or ADD DECOMPOSITION | |
905 1- Parent Code | |
906 2- <Child Code\ [Factor]\ [Yield]> | |
907 """ | |
908 # _____number of fields_____ | |
909 # Any INFORMATION after last field separator is ignored | |
910 # The record must have 3 fields | |
911 if len(field_list) > 3: | |
912 field_list = field_list[:3] | |
913 # If there are no sufficient fields, the fields are added | |
914 # with empty value:"" | |
915 else: | |
916 field_list = field_list + [""]*(3-len(field_list)) | |
917 # control character are erased: end of line, tab, space | |
918 # _____Fields_____ | |
919 _record_type = field_list[0] | |
920 _code = self.delete_control_space(field_list[1]) | |
921 _children = self.delete_control_space(field_list[2]) | |
922 # _____Code_____ | |
923 # "#" and "##" characters at the end of the code are erased | |
924 # invalid characters are also erased | |
925 _code = self.validateCode(_code) | |
926 # _____children_____ | |
927 # TODO: test the number of decimals in factor an yield values | |
928 _children = _children.split( "\\" ) | |
929 _children_list = [ ] | |
930 _child_index = 0 | |
931 while _child_index < len(_children)-3: | |
932 # _____subfields_____ | |
933 _child_code = _children[_child_index] | |
934 _factor = _children[_child_index+1] | |
935 _yield = _children[_child_index+2] | |
936 # _____child_code_____ | |
937 _child_code = self.validateCode(_child_code) | |
938 # _____factor_____ | |
939 if _factor != "": | |
940 try: | |
941 _factor = float(_factor) | |
942 except ValueError: | |
943 print utils.mapping(_("ValueError loadig the "\ | |
944 "descomposition of the record $1, the factor "\ | |
945 "of the child $2 must be a float number and "\ | |
946 "can not be $3, seted default value 1.0"), | |
947 (_code, _child_code, _factor)) | |
948 _factor = 1.0 | |
949 #____yield___ | |
950 if _yield != "": | |
951 try: | |
952 _yield = float(_yield) | |
953 except ValueError: | |
954 print utils.mapping(_("ValueError loading the "\ | |
955 "descomposition of the record $1, the yield of "\ | |
956 "the child $2, must be a float number and can"\ | |
957 "not be $3, seted default value 1.0"), | |
958 (_code, _child_code, _factor)) | |
959 _yield = 1.0 | |
960 if _child_code != "" and _code != "": | |
961 _children_list.append([_child_code, _factor, _yield ]) | |
962 if _record_type == "D": | |
963 _position = _child_index / 3 | |
964 else: #_record_type == "Y" | |
965 _position = -1 | |
966 self.__budget.setTree(_code, _child_code, _position, _factor, | |
967 _yield, "", "", "", "") | |
968 _child_index = _child_index + 3 | |
969 self.num_valid_record = self.num_valid_record +1 | |
970 | |
971 def _parseT(self, field_list): | |
972 """_parseT(self, field_list) | |
973 | |
974 field_list: field list of the record | |
975 0- T: Text | |
976 1- Record code | |
977 2- Description text | |
978 """ | |
979 # _____Number of fields_____ | |
980 # Any INFORMATION after last field separator is ignored | |
981 # The record must have 3 fields | |
982 if len(field_list) > 3: | |
983 field_list = field_list[0:3] | |
984 field_list = field_list[1:] | |
985 if len(field_list) != 2: | |
986 return | |
987 # control character are erased: end of line, tab, space | |
988 # _____Fields_____ | |
989 _code = self.delete_control_space(field_list[0]) | |
990 _text = field_list[1] | |
991 # _____Code_____ | |
992 # "#" and "##" characters at the end of the code are erased | |
993 # invalid characters are also erased | |
994 _code = self.validateCode(_code) | |
995 # _____Text_____ | |
996 self.__budget.setText(_code, _text) | |
997 self.num_valid_record = self.num_valid_record + 1 | |
998 | |
999 def _parseMN(self, field_list): | |
1000 """_parseMN(self, field_list) | |
1001 | |
1002 field_list: field list of the record | |
1003 0- M or N: MEASURE or ADD MEASURE | |
1004 1- [Parent Code\]Child Code | |
1005 2- {Path\} | |
1006 3- TOTAL MEASURE | |
1007 4- {Type\Comment\Unit\Length\Width\Height\} | |
1008 5- [Label] | |
1009 """ | |
1010 | |
1011 # _____Number of fields_____ | |
1012 # Any INFORMATION after last field separator is ignored | |
1013 # The record must have 6 fields | |
1014 if len(field_list) > 6: | |
1015 field_list = field_list[:6] | |
1016 # If there are no sufficient fields, the fields are added | |
1017 # with empty value:"" | |
1018 else: | |
1019 field_list = field_list + [""]*(6-len(field_list)) | |
1020 # control character are erased: end of line, tab, space | |
1021 # _____Fields_____ | |
1022 _record_type = field_list[0] | |
1023 _codes = self.delete_control_space(field_list[1]) | |
1024 _path = self.delete_control_space(field_list[2]) | |
1025 _total = self.delete_control_space(field_list[3]) | |
1026 _lines = self.delete_control(field_list[4]) | |
1027 _label = self.delete_control_space(field_list[5]) | |
1028 # _____Codes_____ | |
1029 _code_list = _codes.split( "\\" ) | |
1030 # "#" and "##" characters at the end of the code are erased | |
1031 # invalid characters are also erased | |
1032 if len(_code_list) == 2: | |
1033 _parent_code = self.validateCode(_code_list[0]) | |
1034 if _parent_code == "": | |
1035 _parent_code = None | |
1036 _child_code = self.validateCode(_code_list[1]) | |
1037 elif len(_code_list) == 1: | |
1038 _child_code = self.validateCode(_code_list[0]) | |
1039 _parent_code = None | |
1040 else: | |
1041 print utils.mapping(_("Invalid codes in $1 record, codes $2"), | |
1042 (_record_type, _codes)) | |
1043 return | |
1044 if _child_code == "": | |
1045 print utils.mapping(_("Empty child code in $1 record, codes: "\ | |
1046 "$2"), (_record_type, _codes)) | |
1047 return | |
1048 # _____Path_____ | |
1049 # TODO: path=0, no-estructured measures | |
1050 _path_list = _path.split( "\\" ) | |
1051 if len(_path_list) > 0: | |
1052 while _path_list[-1] == "": | |
1053 _path_list = _path_list[:-1] | |
1054 _path = _path_list[-1] | |
1055 try: | |
1056 _path = int(_path) | |
1057 except ValueError: | |
1058 print utils.mapping(_("Invalid path in $1 record, "\ | |
1059 "codes $2"), (_record_type, _codes)) | |
1060 return | |
1061 if _path > 0: | |
1062 _path -= 1 | |
1063 else: | |
1064 _path = 0 | |
1065 # _____Total_____ | |
1066 try: | |
1067 _total = float(_total) | |
1068 except ValueError: | |
1069 print utils.mapping(_("Invalid Total Measure value in $1 "\ | |
1070 "record, codes $2"), (_record_type, _codes)) | |
1071 return | |
1072 # _____Measure lines_____ | |
1073 _lines = _lines.split( "\\" ) | |
1074 _line_index = 0 | |
1075 _line_list = [ ] | |
1076 while _line_index < len(_lines)-6: | |
1077 _linetype = _lines[_line_index] | |
1078 if _linetype == "": | |
1079 _linetype = 0 | |
1080 elif _linetype == "1" or _linetype == "2" or \ | |
1081 _linetype == "3": | |
1082 _linetype = int(_linetype) | |
1083 else: | |
1084 _linetype = 0 | |
1085 _comment= _lines[_line_index + 1] | |
1086 if _linetype == 3: | |
1087 # "formula": ".*[^0123456789\.()\+\-\*/\^abcdp ].*" | |
1088 if self.__pattern["formula"].match(_comment): | |
1089 print utils.mapping(_("The comment is not a formula or "\ | |
1090 "its have invalid characters, in the $1 record, "\ | |
1091 "codes $2"), (_record_type, _codes)) | |
1092 return | |
1093 else: | |
1094 _formula = _comment | |
1095 _comment = "" | |
1096 else: | |
1097 _formula = "" | |
1098 _units = _lines[_line_index + 2] | |
1099 _length = _lines[_line_index + 3] | |
1100 _width = _lines[_line_index + 4] | |
1101 _height = _lines[_line_index + 5] | |
1102 try: | |
1103 if _units != "": _units = float(_units) | |
1104 if _length != "": _length = float(_length) | |
1105 if _width != "": _width = float(_width) | |
1106 if _height != "": _height = float(_height) | |
1107 except ValueError: | |
1108 print utils.mapping("The measure values are not float "\ | |
1109 "numbers, code $1", (_codes,)) | |
1110 return | |
1111 _line_list.append([_linetype, _comment, _units, | |
1112 _length, _width, _height, _formula]) | |
1113 _line_index = _line_index + 6 | |
1114 self.__budget.setTree(_parent_code, _child_code, _path, "", "", | |
1115 _total, _line_list, _label, _record_type) | |
1116 self.num_valid_record = self.num_valid_record + 1 | |
1117 | |
1118 def _parseW(self, field_list): | |
1119 """_parseW(self, field_list) | |
1120 | |
1121 field_list: field list of the record | |
1122 0- W: Geografical field | |
1123 1- Field Code | |
1124 2- Field | |
1125 """ | |
1126 # _____Number of fields_____ | |
1127 # Any INFORMATION after last field separator is ignored | |
1128 # The record must have 2 fields | |
1129 if len(field_list) >= 2: | |
1130 field_list = field_list[1:2] | |
1131 else: | |
1132 return | |
1133 # control character are erased: end of line, tab, space | |
1134 # _____Fields_____ | |
1135 _code_fields = field_list[0] | |
1136 # last \ is erased | |
1137 if len(_code_fields) and _code_fields[-1] == "\\": | |
1138 _code_fields = _code_fields[:-1] | |
1139 _code_fields = _code_fields.split("\\") | |
1140 _field_dict = {} | |
1141 _field_index = 0 | |
1142 while _field_index < len(_code_fields)-1: | |
1143 # _____subfields_____ | |
1144 _field_code = _code_fields[_field_index] | |
1145 _field_title = _code_fields[_field_index+1] | |
1146 # control character are erased: end of line, tab, space | |
1147 # _____section_code_____ | |
1148 #"control": "[\t \n\r]" | |
1149 _field_code = self.delete_control_space(_field_code) | |
1150 # _____section_title_____ | |
1151 if _field_code != "": | |
1152 _field_dict[_field_code] = _field_title | |
1153 _field_index = _field_index + 2 | |
1154 self.__budget.setSheetFields(_field_dict) | |
1155 self.num_valid_record = self.num_valid_record +1 | |
1156 | |
1157 def _parseL(self, field_list): | |
1158 """_parseL(self, field_list) | |
1159 | |
1160 field_list: field list of the record | |
1161 0- L: Sheet of Conditions 1 | |
1162 A: | |
1163 1- Empty | |
1164 2- {Section Code\Section Title} | |
1165 B: | |
1166 1- Record Code | |
1167 2- {Section Code\Section Text} | |
1168 3- {Section Code\RTF file} | |
1169 4- {Section Code\HTM file} | |
1170 """ | |
1171 # _____Number of fields_____ | |
1172 # The record must have at least 3 fields | |
1173 if len(field_list) < 3: | |
1174 return | |
1175 _code = field_list[1] | |
1176 if _code == "": | |
1177 # A: Section Titles | |
1178 # Any INFORMATION after last field separator is ignored | |
1179 # The record must have 3 fields | |
1180 if len(field_list) > 3: | |
1181 field_list = field_list[0:3] | |
1182 field_list = field_list[1:3] | |
1183 # _____Fields_____ | |
1184 _section_codes = field_list[1] | |
1185 # last \ is erased | |
1186 if len(_section_codes) and _section_codes[-1] == "\\": | |
1187 _section_codes = _section_codes[:-1] | |
1188 _section_codes = _section_codes.split("\\") | |
1189 _section_dict = {} | |
1190 _section_index = 0 | |
1191 while _section_index < len(_section_codes)-1: | |
1192 # _____subfields_____ | |
1193 _section_code = _section_codes[_section_index] | |
1194 | |
1195 _section_title = _section_codes[_section_index+1] | |
1196 # control character are erased: end of line, tab, space | |
1197 # _____section_code_____ | |
1198 _section_code = self.delete_control_space(_section_code) | |
1199 # _____section_title_____ | |
1200 _section_title = self.delete_control_space(_section_title) | |
1201 if _section_code != "": | |
1202 _section_dict[_section_code] = _section_title | |
1203 _section_index = _section_index + 2 | |
1204 self.__budget.setSheetSections(_section_dict) | |
1205 self.num_valid_record = self.num_valid_record +1 | |
1206 | |
1207 else: | |
1208 # Any INFORMATION after last field separator is ignored | |
1209 # The record must have 5 fields | |
1210 if len(field_list) > 5: | |
1211 field_list = field_list[0:5] | |
1212 field_list = field_list[1:] | |
1213 # _____Fields_____ | |
1214 # _____Record Code_____ | |
1215 _record_code = self.delete_control_space(field_list[0]) | |
1216 # "#" and "##" characters at the end of the code are erased | |
1217 # invalid characters are also erased | |
1218 _record_code = self.validateCode(_record_code) | |
1219 _scodes_text = field_list[1] | |
1220 if _scodes_text == "": | |
1221 # TODO: rtf and html files | |
1222 print "Html and rtf files not implemented in ~L record" | |
1223 else: | |
1224 # _____Section-code_Section-text_____ | |
1225 # last \ is erased | |
1226 if len(_scodes_text) and _scodes_text[-1] == "\\": | |
1227 _scodes_text = _scodes_text[:-1] | |
1228 _scodes_text = _scodes_text.split("\\") | |
1229 _paragraph_dict = {} | |
1230 _section_dict = {} | |
1231 _section_index = 0 | |
1232 while _section_index < len(_scodes_text)-1: | |
1233 # _____subfields_____ | |
1234 _section_code = _scodes_text[_section_index] | |
1235 _section_text = _scodes_text[_section_index+1] | |
1236 # control character are erased: end of line, tab, space | |
1237 # _____section_code_____ | |
1238 _section_code = self.delete_control_space(_section_code) | |
1239 # _____section_text_____ | |
1240 if _section_code != "" and _section_text != "": | |
1241 #-# paragraph #-# | |
1242 _paragraph_code = _record_code + _section_code + "*" | |
1243 _paragraph_dict[ _paragraph_code ] = _section_text | |
1244 _section_dict[_section_code] = _paragraph_code | |
1245 _section_index = _section_index + 2 | |
1246 self.__budget.setSheetParagraphs(_paragraph_dict) | |
1247 self.__budget.setSheetRecord(_record_code, "*", _section_dict) | |
1248 self.num_valid_record = self.num_valid_record +1 | |
1249 | |
1250 def _parseQ(self, field_list): | |
1251 """_parseQ(self, field_list) | |
1252 | |
1253 field_list: field list of the record | |
1254 0- Q: Sheet of Conditions 2 | |
1255 1- Record Code | |
1256 2- {Section Code\Paragraph key\{Field key;}\}| | |
1257 """ | |
1258 # _____Number of fields_____ | |
1259 # The record must have at least 3 fields | |
1260 if len(field_list) < 3: | |
1261 return | |
1262 _code = field_list[1] | |
1263 # Any INFORMATION after last field separator is ignored | |
1264 # The record must have 3 fields | |
1265 if len(field_list) > 3: | |
1266 field_list = field_list[0:3] | |
1267 field_list = field_list[1:] | |
1268 # _____Fields_____ | |
1269 # _____Record Code_____ | |
1270 _record_code = self.delete_control_space(field_list[0]) | |
1271 # "#" and "##" characters at the end of the code are erased | |
1272 # invalid characters are also erased | |
1273 _record_code = self.validateCode(_record_code) | |
1274 _scodes_pkey = field_list[1] | |
1275 # last \ is erased | |
1276 if len(_scodes_pkey) and _scodes_pkey[-1] == "\\": | |
1277 _scodes_pkey = _scodes_pkey[:-1] | |
1278 _scodes_pkey = _scodes_pkey.split("\\") | |
1279 _field_dict = {} | |
1280 _section_index = 0 | |
1281 while _section_index < len(_scodes_pkey) -1: | |
1282 # _____subfields_____ | |
1283 _section_code = _scodes_pkey[_section_index] | |
1284 _paragraph_key = _scodes_text[_section_index+1] | |
1285 _field_keys = _scodes_text[_section_index+2] | |
1286 # control character are erased: end of line, tab, space | |
1287 # _____section_code_____ | |
1288 _section_code = self.delete_control_space(_section_code) | |
1289 # _____section_text_____ | |
1290 _paragraph_key = self.delete_control_space(_paragraph_key) | |
1291 # _____Fields keys_____ | |
1292 _field_keys = self.delete_control_space(_field_keys) | |
1293 # last ; is erased | |
1294 if len(_field_keys) and _field_keys[-1] == ";": | |
1295 _field_keys = _field_keys[:-1] | |
1296 _field_keys_list = _scodes_pkey.split(";") | |
1297 for _field_key in _field_keys_list: | |
1298 if _field_key != "" and _section_code != "" and \ | |
1299 _paragraph_key != "": | |
1300 if _field_key in _field_dict: | |
1301 _section_dict = _field_dict[_field_key] | |
1302 else: | |
1303 _section_dict = {} | |
1304 _field_dict[_field_key] = _section_dict | |
1305 _section_dict[_section_code] = _paragraph_code | |
1306 _section_index = _section_index + 3 | |
1307 for _field, _section_dict in _field_dict.iteritems(): | |
1308 self.__budget.setSheetRecord(_record_code, _field, _section_dict) | |
1309 self.num_valid_record = self.num_valid_record +1 | |
1310 | |
1311 def _parseJ(self, field_list): | |
1312 """_parseJ(self, field_list) | |
1313 | |
1314 field_list: field list of the record | |
1315 0- J: Sheet of Conditions 3 | |
1316 1- Paragraph code | |
1317 2- [Paragraph text] | |
1318 3- [RTF file] | |
1319 4- [HTML file] | |
1320 """ | |
1321 # _____Number of fields_____ | |
1322 # The record must have at least 3 fields | |
1323 if len(field_list) < 3: | |
1324 return | |
1325 # Any INFORMATION after last field separator is ignored | |
1326 # The record must have 5 fields | |
1327 if len(field_list) > 5: | |
1328 field_list = field_list[0:5] | |
1329 field_list = field_list[1:] | |
1330 # _____Fields_____ | |
1331 # _____Paragraph code_____ | |
1332 _paragraph_code = self.delete_control_space(field_list[0]) | |
1333 # _____Paragraph text_____ | |
1334 _paragraph_text = field_list[1] | |
1335 if _paragraph_text == "": | |
1336 # TODO: rtf and html files | |
1337 print "Html and rtf files not implemented in ~J record" | |
1338 else: | |
1339 self.__budget.setSheetParagraph(paragraph_code, paragraph_text) | |
1340 self.num_valid_record = self.num_valid_record +1 | |
1341 | |
1342 def _parseG(self, field_list): | |
1343 """_parseG(self, field_list) | |
1344 | |
1345 field_list: field list of the record | |
1346 0- G: Grafic info | |
1347 1- record code | |
1348 2- <grafic_file.ext\> | |
1349 """ | |
1350 # _____Number of fields_____ | |
1351 # The record must have at least 3 fields | |
1352 if len(field_list) < 3: | |
1353 return | |
1354 # Any INFORMATION after last field separator is ignored | |
1355 # The record must have 3 fields | |
1356 if len(field_list) > 3: | |
1357 field_list = field_list[0:3] | |
1358 field_list = field_list[1:] | |
1359 # _____Fields_____ | |
1360 # _____Record Code_____ | |
1361 _record_code = self.delete_control_space(field_list[0]) | |
1362 # "#" and "##" characters at the end of the code are erased | |
1363 # invalid characters are also erased | |
1364 _record_code = self.validateCode(_record_code) | |
1365 # _____Grafic files_____ | |
1366 _grafic_files = self.delete_control(field_list[1]) | |
1367 # _____subfields_____ | |
1368 # last \ is erased | |
1369 if len(_grafic_files) and _grafic_files[-1] == "\\": | |
1370 _grafic_files = _grafic_files[:-1] | |
1371 _grafic_file_list = _grafic_files.split("\\") | |
1372 _tested_grafic_file_list = [] | |
1373 for _grafic_file in _grafic_file_list: | |
1374 _path = os.path.dirname(self.__filename) | |
1375 _grafic_file_path = os.path.join(_path, _grafic_file) | |
1376 if os.path.exists(_grafic_file_path): | |
1377 _tested_grafic_file_list.append(_grafic_file_path) | |
1378 else: | |
1379 _name_ext = os.path.splitext(_grafic_file) | |
1380 _grafic_file_name = _name_ext[0] | |
1381 _grafic_file_ext = _name_ext[1] | |
1382 _grafic_file_name_u = _grafic_file_name.upper() | |
1383 _grafic_file_name_l = _grafic_file_name.lower() | |
1384 _grafic_file_ext_u = _grafic_file_ext.upper() | |
1385 _grafic_file_ext_l = _grafic_file_ext.lower() | |
1386 _uu = _grafic_file_name_u + _grafic_file_ext_u | |
1387 _ul = _grafic_file_name_u + _grafic_file_ext_l | |
1388 _lu = _grafic_file_name_l + _grafic_file_ext_u | |
1389 _ll = _grafic_file_name_l + _grafic_file_ext_l | |
1390 _grafic_file_path_uu = os.path.join(_path, _uu) | |
1391 _grafic_file_path_ul = os.path.join(_path, _ul) | |
1392 _grafic_file_path_lu = os.path.join(_path, _lu) | |
1393 _grafic_file_path_ll = os.path.join(_path, _ll) | |
1394 if os.path.exists(_grafic_file_path_uu): | |
1395 _tested_grafic_file_list.append(_grafic_file_path_uu) | |
1396 elif os.path.exists(_grafic_file_path_ul): | |
1397 _tested_grafic_file_list.append(_grafic_file_path_ul) | |
1398 elif os.path.exists(_grafic_file_path_lu): | |
1399 _tested_grafic_file_list.append(_grafic_file_path_lu) | |
1400 elif os.path.exists(_grafic_file_path_ll): | |
1401 _tested_grafic_file_list.append(_grafic_file_path_ll) | |
1402 else: | |
1403 print utils.mapping(_("The file $1 do not exist"), | |
1404 (_grafic_file_path,)) | |
1405 if len(_grafic_file_list) > 0: | |
1406 for _grafic_file in _tested_grafic_file_list: | |
1407 self.__budget.addFile(_record_code, _grafic_file, "img", "") | |
1408 self.num_valid_record = self.num_valid_record +1 | |
1409 | |
1410 def _parseE(self, field_list): | |
1411 """_parseE(self, field_list) | |
1412 | |
1413 field_list: field list of the record | |
1414 0- E: Company | |
1415 1- company Code | |
1416 2 [ summary ] | |
1417 3- [ name ] | |
1418 4- { [ type ] \ [ subname ] \ [ address ] \ [ postal_code ] | |
1419 \ [ town ] \ [ province ] \ [ country ] \ { phone; } | |
1420 \ { fax; } \ {contact_person; } \ } | |
1421 5- [ cif ] \ [ web ] \ [ email ] \ | |
1422 """ | |
1423 | |
1424 # _____Number of fields_____ | |
1425 # Any INFORMATION after last field separator is ignored | |
1426 # The record must have 6 fields | |
1427 if len(field_list) > 6: | |
1428 field_list = field_list[1:6] | |
1429 # If there are no sufficient fields, the fields are added | |
1430 # with empty value:"" | |
1431 else: | |
1432 field_list = field_list[1:] + [""]*(6-len(field_list)) | |
1433 # _____Fields_____ | |
1434 # _____company Code_____ | |
1435 _company_code = self.delete_control_space(field_list[0]) | |
1436 if _company_code == "": | |
1437 return | |
1438 # _____Summary_____ | |
1439 | |
1440 _sumamary = self.delete_control(field_list[1]) | |
1441 # _____Name_____ | |
1442 _name = self.delete_control(field_list[2]) | |
1443 # _____local_offices_____ | |
1444 _local_offices = self.delete_control(field_list[3]) | |
1445 # _____subfields of local_offices_____ | |
1446 # last \ is erased | |
1447 if len(_local_offices) and _local_offices[-1] == "\\": | |
1448 _local_offices = _local_offices[:-1] | |
1449 _local_offices_list = _local_offices.split("\\") | |
1450 # If there are no sufficent subfields, the subfields are added | |
1451 # whith empty value | |
1452 _nsub = len(_local_offices_list) % 10 | |
1453 if _nsub != 0: | |
1454 _local_offices_list = _local_offices_list + \ | |
1455 [""]*(10-len(field_list)) | |
1456 _local_offices = [] | |
1457 _local_offices_index = 0 | |
1458 while _local_offices_index < len(_local_offices_list)-9: | |
1459 # _____subfields_____ | |
1460 _type = _local_offices_list[_local_offices_index] | |
1461 _subname = _local_offices_list[_local_offices_index+1] | |
1462 _address = _local_offices_list[_local_offices_index+2] | |
1463 _postal_code = _local_offices_list[_local_offices_index+3] | |
1464 _town = _local_offices_list[_local_offices_index+4] | |
1465 _province = _local_offices_list[_local_offices_index+5] | |
1466 _country = _local_offices_list[_local_offices_index+6] | |
1467 _phone = _local_offices_list[_local_offices_index+7] | |
1468 # last ; is erased | |
1469 if len(_phone) and _phone[-1] == ";": | |
1470 _phone = _phone[:-1] | |
1471 _phone_list = _phone.split(";") | |
1472 _fax = _local_offices_list[_local_offices_index+8] | |
1473 # last ; is erased | |
1474 if len(_fax) and _fax[-1] == ";": | |
1475 _fax = _fax[:-1] | |
1476 _fax_list = _fax.split(";") | |
1477 _contact_person = _local_offices_list[_local_offices_index+9] | |
1478 if _type != "" or _subname != "" or _address != "" or \ | |
1479 _postal_code != "" or _town != "" or _province != "" or \ | |
1480 _country != "" or _phone != "" or _fax != "" or \ | |
1481 _contact_person != "": | |
1482 _local_offices.append([_type, _subname, _address, | |
1483 _postal_code, _town, _province, | |
1484 _country, _phone_list, _fax_list, | |
1485 _contact_person]) | |
1486 _local_offices_index = _local_offices_index + 10 | |
1487 # _____cif web email_____ | |
1488 _c_w_e = self.delete_control_space(field_list[4]) | |
1489 # last \ is erased | |
1490 if len(_c_w_e) and _c_w_e[-1] == "\\": | |
1491 _c_w_e = _c_w_e[:-1] | |
1492 _c_w_e_list = _c_w_e.split("\\") | |
1493 # _____subfields_____ | |
1494 # If there are no sufficient fields, the fields are added | |
1495 # with empty value:"" | |
1496 _c_w_e_list = _c_w_e_list + [""]*(3-len(_c_w_e_list)) | |
1497 _cif = _c_w_e_list[0] | |
1498 _web = _c_w_e_list[1] | |
1499 _email = _c_w_e_list[2] | |
1500 self.__budget.setCompany(_company_code, _sumamary, _name, | |
1501 _local_offices, _cif, _web, _email) | |
1502 self.num_valid_record = self.num_valid_record +1 | |
1503 | |
1504 def _parseX(self, field_list): | |
1505 """_parseX(self, field_list) | |
1506 | |
1507 field_list: field list of the record | |
1508 A) | |
1509 0- X: Tecnical information | |
1510 1- Empty | |
1511 2- < TI_Code \ TI_Descitption \ TI_Unit > | |
1512 B) | |
1513 0- X: Tecnical information | |
1514 1- Record_code | |
1515 2- < TI_Code \ TI_value > | |
1516 """ | |
1517 # Tecnical information | |
1518 # The record must have at least 3 fields | |
1519 if len(field_list) < 3: | |
1520 return | |
1521 # Any INFORMATION after last field separator is ignored | |
1522 # The record must have 3 fields | |
1523 if len(field_list) > 3: | |
1524 field_list = field_list[0:3] | |
1525 field_list = field_list[1:] | |
1526 # _____Fields_____ | |
1527 # "control": "[\t \n\r]" | |
1528 _field_1 = self.delete_control_space(field_list[0]) | |
1529 _field_2 = self.delete_control_space(field_list[1]) | |
1530 if _field_1 == "": | |
1531 # A) | |
1532 _field_2_list = _field_2.split("\\") | |
1533 _ti_index = 0 | |
1534 while _ti_index < len(_field_2_list)-3: | |
1535 _ti_code = _field_2_list[_ti_index] | |
1536 _ti_description = _field_2_list[_ti_index+1] | |
1537 _ti_unit = _field_2_list[_ti_index+2] | |
1538 if _ti_code != "": | |
1539 self.__budget.addTecInfo(_ti_code, _ti_description, | |
1540 _ti_unit) | |
1541 _ti_index = _ti_index + 3 | |
1542 else: | |
1543 # B) | |
1544 # "#" and "##" characters at the end of the code are erased | |
1545 # invalid characters are also erased | |
1546 _record_code = self.validateCode(_field_1) | |
1547 _field_2_list = _field_2.split("\\") | |
1548 _ti_index = 0 | |
1549 _ti_dict = {} | |
1550 while _ti_index < len(_field_2_list)-2: | |
1551 _ti_code = _field_2_list[_ti_index] | |
1552 _ti_value = _field_2_list[_ti_index+1] | |
1553 if _ti_code != "" and _ty_value != "": | |
1554 _ti_dict[_ti_code] = _ty_value | |
1555 _ti_index = _ti_index + 2 | |
1556 self.__budget.setTecnicalInformation(_record_code, _ti_dict) | |
1557 self.num_valid_record = self.num_valid_record +1 | |
1558 | |
1559 def _parseF(self, field_list): | |
1560 """_parseF(self, field_list) | |
1561 | |
1562 field_list: field list of the record | |
1563 0- F: Files | |
1564 1- Record code | |
1565 2- { Type \ { Filenames; } \ [Description] } | |
1566 """ | |
1567 | |
1568 # _____Number of fields_____ | |
1569 # The record must have at least 3 fields | |
1570 if len(field_list) < 3: | |
1571 return | |
1572 # Any INFORMATION after last field separator is ignored | |
1573 # The record must have 3 fields | |
1574 if len(field_list) > 3: | |
1575 field_list = field_list[0:3] | |
1576 field_list = field_list[1:] | |
1577 # _____Fields_____ | |
1578 # _____Record Code_____ | |
1579 _record_code = self.delete_control_space(field_list[0]) | |
1580 # "#" and "##" characters at the end of the code are erased | |
1581 # invalid characters are also erased | |
1582 _record_code = self.validateCode(_record_code) | |
1583 # _____Grafic files_____ | |
1584 _files = self.delete_control(field_list[1]) | |
1585 # _____subfields_____ | |
1586 # last \ is erased | |
1587 if len(_files) and _files[-1] == "\\": | |
1588 _files = _files[:-1] | |
1589 _files_list = _files.split("\\") | |
1590 # adding empty subfiels if necesary | |
1591 if len(_files_list)%3 > 0: | |
1592 _files_list.extend[""]*(3 - len(_files_list)%3) | |
1593 _file_index = 0 | |
1594 _tested_files_list = [] | |
1595 while _file_index < len(_files_list)-3: | |
1596 _type = _files_list[_file_index].replace(" ","") | |
1597 ## _types = { | |
1598 ## "0": _("others"), | |
1599 ## "1": _("características técnicas y de fabricación"), | |
1600 ## "2": _("manual de colocación, uso y mantenimiento"), | |
1601 ## "3": _("certificado/s de elementos y sistemas"), | |
1602 ## "4": _("normativa y bibliografía"), | |
1603 ## "5": _("tarifa de precios"), | |
1604 ## "6": _("condiciones de venta"), | |
1605 ## "7": _("carta de colores"), | |
1606 ## "8": _("ámbito de aplicación y criterios selección"), | |
1607 ## "9": _("cálculo de elementos y sistemas"), | |
1608 ## "10": _("presentación, datos generales, objetivos, etc. de "\ | |
1609 ## "empresa"), | |
1610 ## "11": _("certificado/s de empresa"), | |
1611 ## "12": _("obras realizadas")} | |
1612 _types = ["0", "1", "2", "3", "4", "5", "6", "7", "8", "9", "10", | |
1613 "11", "12"] | |
1614 if not _type in _types: | |
1615 _type = "0" | |
1616 _filenames = _files_list[_file_index + 1] | |
1617 _description = _files_list[_file_index + 2] | |
1618 _file_index += 3 | |
1619 if len(_filenames) and _filenames[-1] == ";": | |
1620 _files = _files[:-1] | |
1621 _filenames_list = _files.split(";") | |
1622 _path = os.path.dirname(self.__filename) | |
1623 for _filename in filenames_list: | |
1624 _file_path = os.path.join(_path, _filename) | |
1625 if os.path.exists(_file_path): | |
1626 _tested_files_list.append([_file_path, _type, | |
1627 _description]) | |
1628 else: | |
1629 _name_ext = os.path.splitext(_filename) | |
1630 _file_name = _name_ext[0] | |
1631 _file_ext = _name_ext[1] | |
1632 _file_name_u = _file_name.upper() | |
1633 _file_name_l = _file_name.lower() | |
1634 _file_ext_u = _file_ext.upper() | |
1635 _file_ext_l = _file_ext.lower() | |
1636 _uu = _file_name_u + _file_ext_u | |
1637 _ul = _file_name_u + _file_ext_l | |
1638 _lu = _file_name_l + _file_ext_u | |
1639 _ll = _file_name_l + _file_ext_l | |
1640 _file_path_uu = os.path.join(_path, _uu) | |
1641 _file_path_ul = os.path.join(_path, _ul) | |
1642 _file_path_lu = os.path.join(_path, _lu) | |
1643 _file_path_ll = os.path.join(_path, _ll) | |
1644 if os.path.exists(_file_path_uu): | |
1645 _tested_files_list.append([_file_path_uu, _type, | |
1646 _description]) | |
1647 elif os.path.exists(_grafic_file_path_ul): | |
1648 _tested_files_list.append([_file_path_ul, _type, | |
1649 _description]) | |
1650 elif os.path.exists(_grafic_file_path_lu): | |
1651 _tested_files_list.append([_file_path_lu, _type, | |
1652 _description]) | |
1653 elif os.path.exists(_grafic_file_path_ll): | |
1654 _tested_files_list.append([_file_path_ll, _type, | |
1655 _description]) | |
1656 else: | |
1657 print utils.mapping(_("The file $1 do not exist"), | |
1658 (_file_path,)) | |
1659 if len(_tested_files_list) > 0: | |
1660 for _file in _tested_file_list: | |
1661 self.__budget.addFile(_record_code, _file[0], file[1], file[2]) | |
1662 self.num_valid_record = self.num_valid_record +1 | |
1663 | |
1664 def _parseB(self, field_list): | |
1665 """_parseB(self, field_list) | |
1666 | |
1667 field_list: field list of the record | |
1668 0- B: Change code | |
1669 1- Record Code | |
1670 2- New code | |
1671 """ | |
1672 # _____Number of fields_____ | |
1673 # Any INFORMATION after last field separator is ignored | |
1674 # The record must have 3 fields | |
1675 if len(field_list) > 3: | |
1676 field_list = field_list[0:3] | |
1677 field_list = field_list[1:] | |
1678 if len(field_list) != 2: | |
1679 return | |
1680 # control character are erased: end of line, tab, space | |
1681 # _____Fields_____ | |
1682 _code = self.delete_control_space(field_list[0]) | |
1683 _new_code = self.delete_control_space(field_list[1]) | |
1684 # _____Codes_____ | |
1685 # "#" and "##" characters at the end of the code are erased | |
1686 # invalid characters are also erased | |
1687 _code = self.validateCode(_code) | |
1688 _new_code = self.validateCode(_new_code) | |
1689 # change code | |
1690 self.__budget.changeCode(_code, _new_code) | |
1691 self.num_valid_record = self.num_valid_record + 1 | |
1692 | |
1693 def _parseA(self, field_list): | |
1694 """_parseA(self, field_list) | |
1695 | |
1696 field_list: field list of the record | |
1697 0- A: Labels | |
1698 1- Record Code | |
1699 2- <Label\> | |
1700 """ | |
1701 # _____Number of fields_____ | |
1702 # Any INFORMATION after last field separator is ignored | |
1703 # The record must have 3 fields | |
1704 if len(field_list) > 3: | |
1705 field_list = field_list[0:3] | |
1706 field_list = field_list[1:] | |
1707 if len(field_list) != 2: | |
1708 return | |
1709 # control character are erased: end of line, tab, space | |
1710 # _____Fields_____ | |
1711 # "control": "[\t \n\r]" | |
1712 _code = self.delete_control_space(field_list[0]) | |
1713 _labels = self.delete_control_space(field_list[1]) | |
1714 # _____Codes_____ | |
1715 # "#" and "##" characters at the end of the code are erased | |
1716 # invalid characters are also erased | |
1717 _code = self.validateCode(_code) | |
1718 # _____Labels_____ | |
1719 # last \ is erased | |
1720 # TODO: change the others parsers to this: | |
1721 while len(_labels) > 0 and _labels[-1] == "\\": | |
1722 _labels = _labels[:-1] | |
1723 # replace "_" to " " | |
1724 _labels = _labels.replace("_"," ") | |
1725 _label_list = _labels.split("\\") | |
1726 for _label in _label_list: | |
1727 self.__budget.addLabel(_code, _label) | |
1728 self.num_valid_record = self.num_valid_record + 1 | |
1729 | |
1730 def _parseP(self, field_list): | |
1731 """_parseP(self, field_list) | |
1732 | |
1733 field_list: Parametric record | |
1734 A) Global paremetric record | |
1735 0- P: Parametric | |
1736 1- Empty | |
1737 2- [Parametric description] | |
1738 3- [library.DLL] | |
1739 B) Family Parametric record | |
1740 0- P: Parametric | |
1741 1- Family Code | |
1742 2- [Parametric description] | |
1743 """ | |
1744 # TODO: Use global parametric record | |
1745 if len(field_list) > 2: | |
1746 # delete control caracters and spaces | |
1747 _family_code = self.delete_control_space(field_list[1]) | |
1748 if _family_code == "": # A)Global paremetric record | |
1749 # The record must have 3 or 4 fields | |
1750 if len(field_list) > 4: | |
1751 field_list = field_list[0:4] | |
1752 field_list = field_list[1:] | |
1753 if len(field_list) == 2: | |
1754 field_list.append("") | |
1755 if len(field_list) != 3: | |
1756 return | |
1757 else: # B)Family Parametric record | |
1758 # The record must have 3 fields | |
1759 if len(field_list) > 3: | |
1760 field_list = field_list[0:3] | |
1761 field_list = field_list[1:] | |
1762 if len(field_list) != 2: | |
1763 print _("PyArq hates parametric DLLs") | |
1764 return | |
1765 else: | |
1766 return | |
1767 # _____Description_____ | |
1768 _description = field_list[1] | |
1769 if _description == "": | |
1770 print _("PyArq hates parametric DLLs") | |
1771 return | |
1772 # Adding last end of line | |
1773 _description = _description + "\r\n" | |
1774 # Delete comments | |
1775 # "comment" : "#.*\r\n" | |
1776 _description = self.__pattern["comment"].sub("\r\n",_description) | |
1777 # Tabs to spaces | |
1778 _description = _description.replace("\t"," ") | |
1779 # Delete empty lines | |
1780 # "empty_line": r"(\r\n) *\r\n" | |
1781 while self.__pattern["empty_line"].search(_description): | |
1782 _description = self.__pattern["empty_line"].sub( | |
1783 lambda x: x.groups()[0], _description) | |
1784 # Delete spaces before and after / | |
1785 # "space_before_backslash" : r"( )+\\" | |
1786 _description = self.__pattern["space_before_backslash"].sub( | |
1787 r"\\",_description) | |
1788 # "space_after_backslash" : r"\\( )+" | |
1789 _description = self.__pattern["space_after_backslash"].sub( | |
1790 r"\\",_description) | |
1791 # Join lines that start but not end with / | |
1792 _description = "\r\n" + _description # add leading end of line | |
1793 # "start_noend_backslash": "(\r\n\\\.*[^\\\])\r\n" | |
1794 while self.__pattern["start_noend_backslash"].search(_description): | |
1795 _description = self.__pattern["start_noend_backslash"].sub( | |
1796 lambda x: x.groups()[0], _description) | |
1797 # Join lines that end with a + - * / ^ and @ & < > <= >= = <> ! | |
1798 # "end_oper" : "(\+|-|\*|/|/^|@|&|<|>|<=|>=|=|!) *\r\n" | |
1799 _description = self.__pattern["end_oper"].sub( | |
1800 lambda x: x.groups()[0], _description) | |
1801 # Join lines for matricial vars | |
1802 # matricial_var : "(\r\n *[%|\$][A-ZÑ].*=.*,) *\r\n" | |
1803 while self.__pattern["matricial_var"].search(_description): | |
1804 _description = self.__pattern["matricial_var"].sub( | |
1805 lambda x: x.groups()[0], _description) | |
1806 _description = _description[2:] # remove leading end of line | |
1807 #_description = re.sub(r"\\( )+",r"\\",_description) | |
1808 _lines = _description.split("\r\n") | |
1809 _final_description = "" | |
1810 _pass_line = 0 | |
1811 for index in range(len(_lines)): | |
1812 _line = _lines[index] | |
1813 # Parse lines | |
1814 if len(_line) != 0: # Delete empty lines | |
1815 if _pass_line > 0: | |
1816 _pass_line = _pass_line -1 | |
1817 _line = "" | |
1818 elif _line.isspace(): | |
1819 _line = "" | |
1820 elif _line[0] != "\\": | |
1821 # Delete spaces out "" delimiter | |
1822 _list = _line.split('"') | |
1823 _final_line = "" | |
1824 for index1 in range(len(_list)): | |
1825 if index1 % 2 != 0: | |
1826 _parcial_line = '"' + _list[index1] | |
1827 else: | |
1828 _parcial_line = '"' + _list[index1].replace(" ","") | |
1829 _final_line = _final_line + _parcial_line | |
1830 _line = _final_line[1:] | |
1831 _lines[index] = _line | |
1832 # parse data | |
1833 if len(_line) > 2 and _line[:2] == "::": | |
1834 # Delete spaces out " delimiter | |
1835 #print "__PRECIO__" + _line[2:] | |
1836 pass | |
1837 elif len(_line) > 2 and _line[:2] == "%:": | |
1838 # Delete spaces out " delimiter | |
1839 #print "__%AUX__" + _line[2:] | |
1840 pass | |
1841 elif len(_line) > 3 and _line[:2] == "%%:": | |
1842 # Delete spaces out " delimiter | |
1843 #print "__%%AUX__" + _line[2:] | |
1844 pass | |
1845 elif self.__pattern["var"].search(_line): | |
1846 # Delete spaces out " delimiter | |
1847 #print "line =", _line | |
1848 while _line.count('"') % 2 == 1 and \ | |
1849 index + _pass_line + 1 < len(_lines) -1: | |
1850 _line = _line + _lines[index + _pass_line + 1] | |
1851 _pass_line = _pass_line + 1 | |
1852 _search = self.__pattern["var"].search(_line) | |
1853 if _search is not None: | |
1854 _var = _search.groups()[0] + " = " + _search.groups()[1] | |
1855 #print "__VAR__" + str(_var) | |
1856 pass | |
1857 else: | |
1858 #print "no __VAR__", _line | |
1859 pass | |
1860 elif self.__pattern["descomposition"].search(_line): | |
1861 # Delete spaces out " delimiter | |
1862 #_patern = "(^[^:]*):(.*)$" | |
1863 _search = self.__pattern["descomposition"].search(_line) | |
1864 if _search is not None: | |
1865 _var = _search.groups()[0] + ":" + _search.groups()[1] | |
1866 #print "__Descomposición__" + str(_var) | |
1867 pass | |
1868 else: | |
1869 #print "no __Descomposición__", _line | |
1870 pass | |
1871 else: | |
1872 print "Parametric: code: " + _family_code | |
1873 print "******* Desconocido *** : " + _line | |
1874 if index-10 > 0: print "-11 :", _lines[index-11] | |
1875 if index-10 > 0: print "-10 :", _lines[index-10] | |
1876 if index-9 > 0: print "-9 :", _lines[index-9] | |
1877 if index-8 > 0: print "-8 :", _lines[index-8] | |
1878 if index-7 > 0: print "-7 :", _lines[index-7] | |
1879 if index-6 > 0: print "-6 :", _lines[index-6] | |
1880 if index-5 > 0: print "-5 :", _lines[index-5] | |
1881 if index-4 > 0: print "-4 :", _lines[index-4] | |
1882 if index-3 > 0: print "-3 :", _lines[index-3] | |
1883 if index-2 > 0: print "-2 :", _lines[index-2] | |
1884 if index-1 > 0: print "-1 :", _lines[index-1] | |
1885 print "-0 :", _lines[index-0] | |
1886 pass | |
1887 else: | |
1888 _parameter_list = _line.split("\\")[1:-1] | |
1889 if len(_parameter_list) >= 2: | |
1890 if _parameter_list[0] == "C" or \ | |
1891 _parameter_list[0] == "COMENTARIO": | |
1892 #print "__COMENTARIO__" + _parameter_list[1] | |
1893 self.__budget.setParametricSelectComment( | |
1894 _family_code, _parameter_list[1]) | |
1895 elif _parameter_list[0] == "R" or \ | |
1896 _parameter_list[0] == "RESUMEN": | |
1897 #print "__RESUMEN__" + _parameter_list[1] | |
1898 self.__budget.setParametricSummary(_family_code, | |
1899 _parameter_list[1]) | |
1900 elif _parameter_list[0] == "T" or \ | |
1901 _parameter_list[0] == "TEXTO": | |
1902 #print "__TEXTO__" + _parameter_list[1] | |
1903 self.__budget.setParametricText(_family_code, | |
1904 _parameter_list[1]) | |
1905 elif _parameter_list[0] == "P" or \ | |
1906 _parameter_list[0] == "PLIEGO": | |
1907 #print "__PLIEGO__" + str(_parameter_list[1:]) | |
1908 pass | |
1909 elif _parameter_list[0] == "K" or \ | |
1910 _parameter_list[0] == "CLAVES": | |
1911 #print "__CLAVES__" + str(_parameter_list[1:]) | |
1912 pass | |
1913 elif _parameter_list[0] == "F" or \ | |
1914 _parameter_list[0] == "COMERCIAL": | |
1915 #print "__COMERCIAL__" + str(_parameter_list[1:]) | |
1916 pass | |
1917 else: | |
1918 #print "==PARAMETRO==" + str(_parameter_list[:]) | |
1919 pass | |
1920 _final_description = _final_description + _line + "\r\n" | |
1921 | |
1922 #print _line | |
1923 # Delete last empty line | |
1924 _description = _final_description[:-2] | |
1925 _lines = _description.split("\r\n") | |
1926 for _line in _lines: | |
1927 pass | |
1928 #print _line | |
1929 self.num_valid_record = self.num_valid_record + 1 | |
1930 | |
1931 def readFile(self, budget=None, filename=None, interface=None): | |
1932 """readFile(self, budget=None, filename=None) | |
1933 | |
1934 filename: the filename of the fiebdc file | |
1935 budget: base.obra object | |
1936 interface: a object to send messages | |
1937 must have printf(message) progress(percent) | |
1938 recordStatistics(...) | |
1939 Return the budget objetc or None if the file can be readed | |
1940 """ | |
1941 if filename != None and budget != None: | |
1942 self.__filename = filename | |
1943 self.__budget = budget | |
1944 self.__budget.filename = self.__filename | |
1945 if self.__filename is None or self.__budget is None or \ | |
1946 self.__cancel == True: | |
1947 return None | |
1948 if not os.path.exists(self.__filename): | |
1949 return None | |
1950 _time = time.time() | |
1951 try: | |
1952 _file = open(self.__filename, 'r') | |
1953 except IOError: | |
1954 print utils.mapping("IOError: $1", (self.__filename,)) | |
1955 return None | |
1956 self.__budget.filename = self.__filename | |
1957 self._record_number = 0 | |
1958 self.num_valid_record = 0 | |
1959 self._record_V_number = 0 | |
1960 self._record_C_number = 0 | |
1961 self._record_D_number = 0 | |
1962 self._record_Y_number = 0 | |
1963 self._record_M_number = 0 | |
1964 self._record_N_number = 0 | |
1965 self._record_T_number = 0 | |
1966 self._record_K_number = 0 | |
1967 self._record_W_number = 0 | |
1968 self._record_L_number = 0 | |
1969 self._record_Q_number = 0 | |
1970 self._record_J_number = 0 | |
1971 self._record_G_number = 0 | |
1972 self._record_E_number = 0 | |
1973 self._record_O_number = 0 | |
1974 self._record_P_number = 0 | |
1975 self._record_X_number = 0 | |
1976 self._record_B_number = 0 | |
1977 self._record_F_number = 0 | |
1978 self._record_A_number = 0 | |
1979 self._record_Unknow_number = 0 | |
1980 print utils.mapping(_("Loading file $1"), (self.__filename,)) | |
1981 _filesize = float(os.path.getsize(self.__filename)) | |
1982 interface.progress(_file.tell() / _filesize) | |
1983 _buffer = _file.read(1000) | |
1984 # set codepage from V record | |
1985 _record_list = _buffer.split("~") | |
1986 registro_V = _record_list[1] | |
1987 # ~V|[PROPIEDAD_ARCHIVO]|VERSION_FORMATO[\DDMMAAAA]|[PROGRAMA_EMISION]| | |
1988 # [CABECERA]\{ ROTULO_IDENTIFICACION \}|[JUEGO_CARACTERES]| | |
1989 # [COMENTARIO]|[TIPO INFORMACIÓN]|[NÚMERO CERTIFICACIÓN]| | |
1990 # [FECHA CERTIFICACIÓN ] | | |
1991 registro_V = registro_V.split("|") | |
1992 if registro_V[0] == "V": | |
1993 #_codepage = registro_V[5] | |
1994 if len(registro_V) > 5: | |
1995 _version = registro_V[5].strip() | |
1996 # remove leading spaces | |
1997 if _version in self.__character_sets_dict: | |
1998 self.__character_set = self.__character_sets_dict[_version] | |
1999 else: | |
2000 print utils.mapping(_("This codepage do not exist in "\ | |
2001 "FIEBDC3! Default codepage: $1"), | |
2002 (self.__character_set,)) | |
2003 else: | |
2004 print utils.mapping(_("This V record dot have a codepage! "\ | |
2005 "Default codepage: $1"), | |
2006 (self.__character_set,)) | |
2007 else: | |
2008 print utils.mapping(_("Not 'V' record in File! Default codepage: "\ | |
2009 "$1"), (self.__character_set,)) | |
2010 if self.__character_set != "utf8": | |
2011 _buffer = unicode(_buffer, self.__character_set) | |
2012 _buffer = _buffer.encode("utf8") | |
2013 # Any INFORMATION between the beginning of the file and the | |
2014 # beginning of the first registry “~” is ignored | |
2015 #"after_first_tilde" : "^[^~]*~" | |
2016 _buffer = self.__pattern["after_first_tilde"].sub("",_buffer) | |
2017 while _buffer != "" and self.__cancel != True: | |
2018 #-# the blank characters (32), tabs (9) and end of line (13 and 10) | |
2019 # before the separators '~', '|' are erased. | |
2020 # Before separator \ not deleted because it affects the reading of | |
2021 # the record ~P | |
2022 _buffer = self.eraseControlCharacters(_buffer) | |
2023 _record_list = _buffer.split("~") | |
2024 # The last record can be incomplete unless it is the last one of | |
2025 # the file | |
2026 if len(_record_list) > 1: | |
2027 # not the end | |
2028 _last_record = _record_list.pop() | |
2029 else: | |
2030 # the end record | |
2031 # The blank characters (32), tabs (9) and end of line | |
2032 # (13 and 10) at the end of the file are ignored. | |
2033 #"end_control" : "((\r\n)| |\t)+$" | |
2034 _record_list[-1] = self.__pattern["end_control"].sub("", | |
2035 _record_list[-1]) | |
2036 _last_record = "" | |
2037 for record in _record_list: | |
2038 if self.__cancel == True: | |
2039 break | |
2040 self.parseRecord(record) | |
2041 interface.progress(_file.tell() / _filesize) | |
2042 _buffer2 = _file.read(100000) | |
2043 if self.__character_set != "utf8": | |
2044 _buffer2 = unicode(_buffer2, self.__character_set) | |
2045 _buffer2 = _buffer2.encode("utf8") | |
2046 _buffer = _last_record + _buffer2 | |
2047 _file.close() | |
2048 if self.__cancel == True: | |
2049 print _("Cancelled process") | |
2050 return None | |
2051 else: | |
2052 print utils.mapping(_("Time to load: $1 seconds"), | |
2053 (("%.2f" %(time.time()-_time)),)) | |
2054 print utils.mapping(_("Records/Valid Records: $1/$2"), | |
2055 (self._record_number, self.num_valid_record)) | |
2056 if self._record_O_number > 0: | |
2057 print utils.mapping(_("$1 unsuported record type O: "\ | |
2058 "Comercial Relationship"), (self._record_O_number,)) | |
2059 if self.num_valid_record == 0: | |
2060 print _("This file is not a valid FIBDC3 file") | |
2061 return None | |
2062 _str = "" | |
2063 for type in \ | |
2064 [("V", self._record_V_number), | |
2065 ("C", self._record_C_number), | |
2066 ("D", self._record_D_number), | |
2067 ("Y", self._record_Y_number), | |
2068 ("M", self._record_M_number), | |
2069 ("N", self._record_N_number), | |
2070 ("T", self._record_T_number), | |
2071 ("K", self._record_K_number), | |
2072 ("W", self._record_W_number), | |
2073 ("L", self._record_L_number), | |
2074 ("Q", self._record_Q_number), | |
2075 ("J", self._record_J_number), | |
2076 ("G", self._record_G_number), | |
2077 ("E", self._record_E_number), | |
2078 ("O", self._record_O_number), | |
2079 ("P", self._record_P_number), | |
2080 ("X", self._record_X_number), | |
2081 ("B", self._record_B_number), | |
2082 ("F", self._record_F_number), | |
2083 ("A", self._record_A_number), | |
2084 ("?", self._record_Unknow_number)]: | |
2085 _str = _str + "%s: %s\n" %(type[0], type[1]) | |
2086 print _str | |
2087 self._testBudget(self.__budget) | |
2088 return self.__budget | |
2089 | |
2090 def _testBudget(self, budget): | |
2091 """testBudget(self,budget) | |
2092 | |
2093 budget: base.obra object | |
2094 Test and repair budget object after read it from bc3 file | |
2095 """ | |
2096 # TODO: more to do here | |
2097 print _("Testing budget ...") | |
2098 # Add price to records without price | |
2099 _iter = budget.iter() | |
2100 _titlelist = budget.getTitleList()[1] | |
2101 if len(_titlelist) == 0: | |
2102 _titlenum = 1 | |
2103 else: | |
2104 _titlenum = len(_titlelist) | |
2105 for _code in _iter: | |
2106 _record = budget.getRecord(_code) | |
2107 _prices = _record.getPrices() | |
2108 _len_prices = len(_prices) | |
2109 if _titlenum > _len_prices: | |
2110 _leftprices = _titlenum - _len_prices | |
2111 for _index in range(0,_leftprices): | |
2112 _root = budget.getRecord(budget.getRoot()) | |
2113 _price = [0.0, _root.getDate(_len_prices + _index)] | |
2114 budget.addPriceToRecord(_price,_record) | |
2115 print _("End Test") | |
2116 | |
2117 def delete_control_space(self, text): | |
2118 text = self.delete_control(text) | |
2119 text = text.replace(" ", "") | |
2120 return text | |
2121 | |
2122 def delete_control(self, text): | |
2123 text = text.replace("\t", "") | |
2124 text = text.replace("\r", "") | |
2125 text = text.replace("\n", "") | |
2126 return text |