1 # -*- coding: utf-8 -*-
2 # CONFIGURATION MANAGEMENT OF EDF VERSION
3 # ======================================================================
4 # COPYRIGHT (C) 1991 - 2002 EDF R&D WWW.CODE-ASTER.ORG
5 # THIS PROGRAM IS FREE SOFTWARE; YOU CAN REDISTRIBUTE IT AND/OR MODIFY
6 # IT UNDER THE TERMS OF THE GNU GENERAL PUBLIC LICENSE AS PUBLISHED BY
7 # THE FREE SOFTWARE FOUNDATION; EITHER VERSION 2 OF THE LICENSE, OR
8 # (AT YOUR OPTION) ANY LATER VERSION.
10 # THIS PROGRAM IS DISTRIBUTED IN THE HOPE THAT IT WILL BE USEFUL, BUT
11 # WITHOUT ANY WARRANTY; WITHOUT EVEN THE IMPLIED WARRANTY OF
12 # MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. SEE THE GNU
13 # GENERAL PUBLIC LICENSE FOR MORE DETAILS.
15 # YOU SHOULD HAVE RECEIVED A COPY OF THE GNU GENERAL PUBLIC LICENSE
16 # ALONG WITH THIS PROGRAM; IF NOT, WRITE TO EDF R&D CODE_ASTER,
17 # 1 AVENUE DU GENERAL DE GAULLE, 92141 CLAMART CEDEX, FRANCE.
20 # ======================================================================
24 import types,traceback,sys,os
25 import string,linecache
30 from Noyau.N_ASSD import ASSD
31 #from Noyau.N_LASSD import LASSD
32 from Noyau.N_ETAPE import ETAPE
33 from Noyau.N_Exception import AsException
34 from Extensions import commentaire,parametre,parametre_eval
41 class JDC(I_OBJECT.OBJECT):
46 self.etapes_niveaux=[]
50 self._etape_context=None
51 self.recorded_units={}
52 self.old_recorded_units={}
54 def get_index(self,objet):
56 Retourne la position d'objet dans la liste self
58 return self.etapes.index(objet)
60 def get_sd_avant_du_bon_type(self,etape,types_permis):
62 Retourne la liste des concepts avant etape d'un type acceptable
64 d=self.get_contexte_avant(etape)
67 if type(v) != types.InstanceType and not isinstance(v,object): continue
68 # On considère que seul assd indique un type quelconque pas CO
69 elif self.assd in types_permis :
71 elif self.est_permis(v,types_permis):
76 def get_sd_avant_du_bon_type_pour_type_de_base(self,etape,type):
78 Retourne la liste des concepts avant etape d'1 type de base acceptable
79 Attention different de la routine précédente : 1 seul type passé en parametre
80 Teste sur issubclass et par sur le type permis
82 d=self.get_contexte_avant(etape)
85 typeverif=self.cata[0].__dict__[type]
89 if issubclass(v.__class__,typeverif):
94 def cherche_list_avant(self,etape,valeur):
95 d=self.get_contexte_avant(etape)
97 if issubclass(v.__class__,LASSD):
100 # Attention pour enlever les . a la fin des pretendus reels
101 if k == valeur[0:-1] :
105 def est_permis(self,v,types_permis):
106 for type_ok in types_permis:
107 if type_ok in ('R','I','C','TXM') and v in self.params :
109 elif type_ok == 'R' and v.__class__.__name__ == 'reel' :
111 elif type_ok == 'I' and v.__class__.__name__ == 'entier' :
113 elif type_ok == 'C' and v.__class__.__name__ == 'complexe' :
115 elif type_ok == 'TXM' and v.__class__.__name__ == 'chaine' :
117 elif type(type_ok) != types.ClassType and not isinstance(type_ok,type):
119 elif v.__class__ == type_ok or issubclass(v.__class__,type_ok):
123 def addentite(self,name,pos):
126 Si name est le nom d une commande ou un commentaire ajoute
128 Sinon remonte une erreur
132 if name == "COMMENTAIRE" :
133 # ajout d'un commentaire
134 self.set_current_step()
136 for child in self.etapes :
137 if isinstance(child,commentaire.COMMENTAIRE):
139 objet = commentaire.COMMENTAIRE('',parent=self)
140 objet.nom = "_comm_"+`ind`
141 if pos == None : pos = 0
142 self.etapes.insert(pos,objet)
145 CONNECTOR.Emit(self,"add",objet)
148 elif name == "PARAMETRE":
149 # ajout d'un parametre
150 self.set_current_step()
151 nom_param = '_param_'+str(len(self.params)+1)
152 objet = parametre.PARAMETRE(nom=nom_param)
153 if pos == None : pos = 0
154 self.etapes.insert(pos,objet)
158 CONNECTOR.Emit(self,"add",objet)
161 elif name == "PARAMETRE_EVAL":
162 # ajout d'un parametre EVAL
163 self.set_current_step()
164 nom_param = '_param_'+str(len(self.params)+1)
165 objet = parametre_eval.PARAMETRE_EVAL(nom=nom_param)
166 if pos == None : pos = 0
167 self.etapes.insert(pos,objet)
171 CONNECTOR.Emit(self,"add",objet)
174 elif type(name)==types.InstanceType:
175 # on est dans le cas où on veut ajouter une commande déjà
176 # existante (par copie donc)
177 # on est donc nécessairement en mode editeur ...
179 # Il ne faut pas oublier de reaffecter le parent d'obj (si copie)
181 self.set_current_step()
182 if isinstance(objet,ETAPE):
183 if objet.nom_niveau_definition == 'JDC':
184 # l'objet dépend directement du JDC
187 # l'étape dépend d'un niveau et non directement du JDC :
188 # il faut l'enregistrer dans le niveau de parent
189 objet.parent.dict_niveaux[objet.nom_niveau_definition].register(objet)
190 objet.niveau = objet.parent.dict_niveaux[objet.nom_niveau_definition]
191 self.etapes.insert(pos,objet)
192 # il faut vérifier que les concepts utilisés par objet existent bien
193 # à ce niveau d'arborescence
194 objet.verif_existence_sd()
195 objet.update_mc_global()
199 CONNECTOR.Emit(self,"add",objet)
203 # On veut ajouter une nouvelle commande
205 self.set_current_step()
206 cmd=self.get_cmd(name)
207 # L'appel a make_objet n'a pas pour effet d'enregistrer l'étape
208 # auprès du step courant car editmode vaut 1
209 # Par contre elle a le bon parent grace a set_current_step
211 if pos == None : pos = 0
212 self.etapes.insert(pos,e)
213 self.reset_current_step()
217 CONNECTOR.Emit(self,"add",e)
220 except AsException,e:
221 self.reset_current_step()
223 raise AsException("Impossible d ajouter la commande "+name + '\n' +str(e))
225 traceback.print_exc()
226 self.reset_current_step()
228 raise AsException("Impossible d ajouter la commande "+name)
231 #print "JDC.close",self
232 for etape in self.etapes:
233 if hasattr(etape,"close"):etape.close()
234 CONNECTOR.Emit(self,"close")
236 def set_current_step(self):
237 CONTEXT.unset_current_step()
238 CONTEXT.set_current_step(self)
240 def reset_current_step(self):
241 CONTEXT.unset_current_step()
243 def liste_mc_presents(self):
246 def get_sd_avant_etape(self,nom_sd,etape):
247 return self.get_contexte_avant(etape).get(nom_sd,None)
249 def get_sd_apres_etape_avec_detruire(self,nom_sd,sd,etape,avec='non'):
251 Cette méthode retourne la SD sd de nom nom_sd qui est éventuellement
252 définie apres etape en tenant compte des concepts detruits
253 Si avec vaut 'non' exclut etape de la recherche
255 #print "JDC.get_sd_apres_etape_avec_detruire",nom_sd,sd
256 ietap=self.etapes.index(etape)
257 if avec == 'non':ietap=ietap+1
259 for e in self.etapes[ietap:]:
262 autre_sd=d.get(nom_sd,None)
264 # Le concept a ete detruit. On interrompt la recherche car il n'y a
265 # pas eu de redefinition du concept (il n'y a pas de conflit potentiel).
267 if autre_sd is not sd :
268 # L'etape produit un concept different de meme nom. La situation n'est
269 # pas saine (sauf peut etre si reuse ???)
270 if hasattr(e,'reuse') and e.reuse == autre_sd:
271 # Le concept etant reutilise, on interrompt la recherche.
272 # On considere qu'il n'y a pas de nouveau concept defini
273 # meme si dans les etapes suivantes le concept est detruit
274 # et un concept de meme nom créé.
275 # AVERIFIER : avec reuse le concept devrait etre le meme
276 # le passage par ici est tres improbable
279 # Le concept est produit par l'etape (Il y a conflit potentiel).
280 # Le concept est redefini par une etape posterieure.
282 # Pas de destruction du concept ni de redefinition. On retourne le
286 def get_sd_apres_etape(self,nom_sd,etape,avec='non'):
288 Cette méthode retourne la SD de nom nom_sd qui est éventuellement
290 Si avec vaut 'non' exclut etape de la recherche
292 ietap=self.etapes.index(etape)
293 if avec == 'non':ietap=ietap+1
294 for e in self.etapes[ietap:]:
295 sd=e.get_sdprods(nom_sd)
297 if hasattr(e,'reuse'):
302 def get_sd_autour_etape(self,nom_sd,etape,avec='non'):
304 Fonction: retourne la SD de nom nom_sd qui est éventuellement
305 définie avant ou apres etape
306 Permet de vérifier si un concept de meme nom existe dans le périmètre
308 Si avec vaut 'non' exclut etape de la recherche
310 sd=self.get_sd_avant_etape(nom_sd,etape)
312 return self.get_sd_apres_etape(nom_sd,etape,avec)
314 def get_contexte_apres(self,etape):
316 Retourne le dictionnaire des concepts connus apres etape
317 On tient compte des commandes qui modifient le contexte
318 comme DETRUIRE ou les macros
319 Si etape == None, on retourne le contexte en fin de JDC
321 if not etape: return self.get_contexte_avant(etape)
323 d=self.get_contexte_avant(etape)
324 if etape.isactif():etape.update_context(d)
325 self.index_etape_courante=self.index_etape_courante+1
328 def active_etapes(self):
330 Cette méthode a pour fonction de désactiver les étapes qui doivent
331 l'être cad, dans le cas d'ASTER, les étapes qui ne sont pas
332 comprises entre le premier DEBUT/POURSUITE et le premier FIN
333 et rendre actives les autres
335 if self.definition.code == 'ASTER' :
336 # Seulement pour ASTER :
337 # Avant DEBUT actif vaut 0
338 # Apres DEBUT et avant le 1er FIN actif vaut 1
339 # Apres le 1er FIN actif vaut -1
343 for etape in self.etapes:
344 if actif == 0 and etape.nom in ['DEBUT','POURSUITE']:actif=1
349 if etape.nom == 'FIN':actif=-1
351 def suppentite(self,etape) :
353 Cette methode a pour fonction de supprimer une étape dans
355 Retourne 1 si la suppression a pu être effectuée,
356 Retourne 0 dans le cas contraire
358 #print "suppentite",self
359 #PN correction de bugs
360 if etape not in self.etapes:
364 index_etape=self.etapes.index(etape)
365 self.etapes.remove(etape)
367 if etape.niveau is not self:
368 # Dans ce cas l'étape est enregistrée dans un niveau
369 # Il faut la désenregistrer
370 etape.niveau.unregister(etape)
372 etape.supprime_sdprods()
377 # Apres suppression de l'etape il faut controler que les etapes
378 # suivantes ne produisent pas des concepts DETRUITS dans op_init de etape
380 index_etape=index_etape-1
381 etape=self.etapes[index_etape]
384 self.control_context_apres(etape)
387 CONNECTOR.Emit(self,"supp",etape)
391 def control_context_apres(self,etape):
393 Cette méthode verifie que les etapes apres l'etape etape
394 ont bien des concepts produits acceptables (pas de conflit de
396 Si des concepts produits ne sont pas acceptables ils sont supprimés.
397 Effectue les verifications sur les etapes du jdc mais aussi sur les
398 jdc parents s'ils existent.
400 #print "control_context_apres",self,etape
401 #Regularise les etapes du jdc apres l'etape etape
402 self.control_jdc_context_apres(etape)
404 def control_jdc_context_apres(self,etape):
406 Methode semblable a control_context_apres mais ne travaille
407 que sur les etapes et sous etapes du jdc
409 #print "control_jdc_context_apres",self,etape
411 # on demarre de la premiere etape
414 index_etape=self.etapes.index(etape)+1
417 etape=self.etapes[index_etape]
419 #derniere etape du jdc : rien a faire
422 context=self.get_contexte_avant(etape)
424 for e in self.etapes[index_etape:]:
425 e.control_sdprods(context)
426 e.update_context(context)
430 if not self.cr.estvide():return
434 def register_parametre(self,param):
436 Cette méthode sert à ajouter un paramètre dans la liste des paramètres
438 self.params.append(param)
440 def register_fonction(self,fonction):
442 Cette méthode sert à ajouter une fonction dans la liste des fonctions
444 self.fonctions.append(fonction)
446 def delete_param(self,param):
448 Supprime le paramètre param de la liste des paramètres
451 if param in self.params : self.params.remove(param)
452 if self.g_context.has_key(param.nom) : del self.g_context[param.nom]
454 def get_parametres_fonctions_avant_etape(self,etape):
456 Retourne deux éléments :
457 - une liste contenant les noms des paramètres (constantes ou EVAL)
459 - une liste contenant les formules définies avant etape
463 # on récupère le contexte avant etape
464 # on ne peut mettre dans les deux listes que des éléments de ce contexte
465 d=self.get_contexte_avant(etape)
466 # construction de l_constantes
467 for param in self.params:
469 if not nom : continue
470 if d.has_key(nom): l_constantes.append(nom)
471 # construction de l_fonctions
472 for form in self.fonctions:
474 if not nom : continue
475 if d.has_key(nom): l_fonctions.append(form.get_formule())
477 # on ajoute les concepts produits par DEFI_VALEUR
478 # XXX On pourrait peut etre faire plutot le test sur le type
479 # de concept : entier, reel, complexe, etc.
480 for k,v in d.items():
481 if hasattr(v,'etape') and v.etape.nom in ('DEFI_VALEUR',):
482 l_constantes.append(k)
484 # on retourne les deux listes
485 return l_constantes,l_fonctions
487 def get_nb_etapes_avant(self,niveau):
489 Retourne le nombre d etapes avant le debut de niveau
492 for niv in self.etapes_niveaux:
493 if niv == niveau:break
494 nb=nb+len(niv.etapes)
497 def send_message(self,message):
499 self.appli.send_message(message)
501 def init_modif(self):
503 Méthode appelée au moment où une modification va être faite afin de
504 déclencher d'éventuels traitements pré-modification
506 #print "init_modif",self
507 self.state = 'modified'
510 #print "fin_modif",self
511 CONNECTOR.Emit(self,"valid")
515 def deep_update_condition_bloc(self):
516 # pour le moment, on ne fait rien
517 raise "Not implemented"
519 def update_condition_bloc(self):
520 # pour le moment, on ne fait rien
521 raise "Not implemented"
523 def get_liste_mc_inconnus(self):
525 Retourne une liste contenant les mots-clés inconnus à la relecture du JDC
527 # cette liste a le format suivant : [etape,(bloc,mcfact,...),nom_mc,valeur_mc]
529 for etape in self.etapes :
531 if not etape.isvalid() :
532 l = etape.get_liste_mc_inconnus()
533 if l : l_mc.extend(l)
536 def get_genealogie(self):
538 Retourne la liste des noms des ascendants de l'objet self
539 jusqu'à la première ETAPE parent.
543 def get_liste_cmd(self):
545 Retourne la liste des commandes du catalogue
547 return self.niveau.definition.get_liste_cmd()
549 def get_groups(self):
551 Retourne la liste des groupes
553 return self.niveau.definition.liste_groupes,self.niveau.definition.dict_groupes
555 def set_etape_context(self,etape):
557 Positionne l'etape qui sera utilisee dans NommerSdProd pour
558 decider si le concept passé pourra etre nommé
560 self._etape_context=etape
562 def reset_context(self):
564 Cette methode reinitialise le contexte glissant pour pouvoir
565 tenir compte des modifications de l'utilisateur : création
566 de commandes, nommage de concepts, etc.
568 #print "reset_context",self,self.nom
569 self.current_context={}
570 self.index_etape_courante=0
571 # for etape in self.etapes:
572 # etape.reset_context()
574 def del_sdprod(self,sd):
576 Supprime la SD sd de la liste des sd et des dictionnaires de contexte
578 #print "del_sdprod",self,sd
579 #print "del_sdprod",self.sds
580 #print "del_sdprod",self.g_context
581 #print "del_sdprod",self.sds_dict
582 if sd in self.sds : self.sds.remove(sd)
583 if self.g_context.has_key(sd.nom) : del self.g_context[sd.nom]
584 if self.sds_dict.has_key(sd.nom) : del self.sds_dict[sd.nom]
586 def del_param(self,param):
588 Supprime le paramètre param de la liste des paramètres
591 if param in self.params : self.params.remove(param)
592 if self.g_context.has_key(param.nom) : del self.g_context[param.nom]
594 def del_fonction(self,fonction):
596 Supprime la fonction fonction de la liste des fonctions
599 if fonction in self.fonctions : self.fonctions.remove(fonction)
600 if self.g_context.has_key(fonction.nom) : del self.g_context[fonction.nom]
602 def append_sdprod(self,sd):
604 Ajoute la SD sd à la liste des sd en vérifiant au préalable qu'une SD de
605 même nom n'existe pas déjà
607 if sd == None or sd.nom == None:return
609 o=self.sds_dict.get(sd.nom,None)
610 if isinstance(o,ASSD):
611 raise AsException("Nom de concept deja defini : %s" % sd.nom)
612 self.sds_dict[sd.nom]=sd
613 self.g_context[sd.nom] = sd
614 if sd not in self.sds : self.sds.append(sd)
616 def append_param(self,param):
618 Ajoute le paramètre param à la liste des params
619 et au contexte global
621 # il faudrait vérifier qu'un paramètre de même nom n'existe pas déjà !!!
622 if param not in self.params : self.params.append(param)
623 self.g_context[param.nom]=param
625 def append_fonction(self,fonction):
627 Ajoute la fonction fonction à la liste des fonctions
628 et au contexte global
630 # il faudrait vérifier qu'une fonction de même nom n'existe pas déjà !!!
631 if fonction not in self.fonctions : self.fonctions.append(fonction)
632 self.g_context[fonction.nom]=fonction
634 def delete_concept(self,sd):
639 Mettre a jour les etapes du JDC suite à la disparition du
641 Seuls les mots cles simples MCSIMP font un traitement autre
642 que de transmettre aux fils
644 #print "delete_concept",self,sd
645 for etape in self.etapes :
646 etape.delete_concept(sd)
648 def replace_concept_after_etape(self,etape,old_sd,sd):
650 Met à jour les étapes du JDC qui sont après etape en fonction
651 du remplacement du concept sd
653 index = self.etapes.index(etape)+1
654 if index == len(self.etapes) :
655 return # etape est la dernière étape du jdc ...on ne fait rien !
656 for child in self.etapes[index:]:
657 child.replace_concept(old_sd,sd)
659 def update_concept_after_etape(self,etape,sd):
661 Met à jour les étapes du JDC qui sont après etape en fonction
662 de la modification (principalement nommage) du concept sd
665 #On traite toutes les etapes
668 index = self.etapes.index(etape)+1
669 if index == len(self.etapes) :
670 return # etape est la dernière étape du jdc ...on ne fait rien !
671 for child in self.etapes[index:]:
672 child.update_concept(sd)
674 def dump_state(self):
675 print "JDC.state: ",self.state
676 for etape in self.etapes :
677 print etape.nom+".state: ",etape.state
679 def change_unit(self,unit,etape,old_unit):
680 #print "change_unit",unit,etape,old_unit
681 #print id(self.recorded_units),self.recorded_units
682 #if self.recorded_units.has_key(old_unit):del self.recorded_units[old_unit]
683 self.record_unit(unit,etape)
685 def record_unit(self,unit,etape):
686 """Enregistre les unites logiques incluses et les infos relatives a l'etape"""
687 #print "record_unit",unit,etape
690 self.recorded_units[None]=(etape.fichier_ini ,etape.fichier_text,etape.recorded_units)
692 self.recorded_units[unit]=(etape.fichier_ini ,etape.fichier_text,etape.recorded_units)
693 #print id(self.recorded_units),self.recorded_units
694 #print self.recorded_units.get(None,(None,"",{}))[2]
695 #print self.recorded_units.get(None,(None,"",{}))[2].get(None,(None,"",{}))
697 def changefichier(self,fichier):
700 def eval_in_context(self,valeur,etape):
701 """ Tente d'evaluer valeur dans le contexte courant de etape
702 Retourne le parametre valeur inchange si l'evaluation est impossible
704 #contexte initial du jdc
705 context=self.condition_context.copy()
706 #contexte courant des concepts. Il contient les parametres
707 context.update(self.get_contexte_avant(etape))
709 objet = eval(valeur,context)
712 #traceback.print_exc()
716 #ATTENTION SURCHARGE : cette methode doit etre gardée en synchronisation avec celle de Noyau
718 #print "supprime",self
719 Noyau.N_JDC.JDC.supprime(self)
722 # self.const_context={}
726 # self.current_context={}
727 # self.condition_context={}
728 # self.etapes_niveaux=[]
732 # self._etape_context=None
735 #ATTENTION SURCHARGE : cette methode doit etre gardée en synchronisation avec celle de Noyau
736 def register(self,etape):
738 Cette méthode ajoute etape dans la liste
739 des etapes self.etapes et retourne l identificateur d'étape
740 fourni par l appel a g_register
742 A quoi sert editmode ?
743 - Si editmode vaut 1, on est en mode edition de JDC. On cherche
744 à enregistrer une étape que l'on a créée avec eficas (en passant
745 par addentite) auquel cas on ne veut récupérer que son numéro
746 d'enregistrement et c'est addentité qui l'enregistre dans
747 self.etapes à la bonne place...
748 - Si editmode vaut 0, on est en mode relecture d'un fichier de
749 commandes et on doit enregistrer l'étape à la fin de self.etapes
750 (dans ce cas l'ordre des étapes est bien l'ordre chronologique
753 if not self.editmode:
754 self.etapes.append(etape)
757 return self.g_register(etape)
759 #ATTENTION SURCHARGE : cette methode doit etre gardée en synchronisation avec celle de Noyau
760 def NommerSdprod(self,sd,sdnom,restrict='non'):
762 Nomme la SD apres avoir verifie que le nommage est possible :
764 Si le nom est deja utilise, leve une exception
765 Met le concept créé dans le concept global g_context
767 # XXX En mode editeur dans EFICAS, le nommage doit etre géré différemment
768 # Le dictionnaire g_context ne représente pas le contexte
769 # effectif avant une étape.
770 # Il faut utiliser get_contexte_avant avec indication de l'étape
772 # Cette etape est indiquee par l'attribut _etape_context qui a ete
773 # positionné préalablement par un appel à set_etape_context
775 if CONTEXT.debug : print "JDC.NommerSdprod ",sd,sdnom
777 if self._etape_context:
778 o=self.get_contexte_avant(self._etape_context).get(sdnom,None)
780 o=self.sds_dict.get(sdnom,None)
782 if isinstance(o,ASSD):
783 raise AsException("Nom de concept deja defini : %s" % sdnom)
785 # ATTENTION : Il ne faut pas ajouter sd dans sds car il s y trouve deja.
786 # Ajoute a la creation (appel de reg_sd).
787 self.sds_dict[sdnom]=sd
790 # En plus si restrict vaut 'non', on insere le concept dans le contexte du JDC
791 if restrict == 'non':
792 self.g_context[sdnom]=sd
794 #ATTENTION SURCHARGE : cette methode doit etre gardée en synchronisation avec celle de Noyau
795 def delete_concept_after_etape(self,etape,sd):
797 Met à jour les étapes du JDC qui sont après etape en fonction
798 de la disparition du concept sd
800 index = self.etapes.index(etape)+1
801 if index == len(self.etapes) :
802 return # etape est la dernière étape du jdc ...on ne fait rien !
803 for child in self.etapes[index:]:
804 child.delete_concept(sd)
806 #ATTENTION SURCHARGE : les methodes ci-dessous surchargent des methodes de Noyau et Validation : a reintegrer
808 def get_file(self,unite=None,fic_origine=''):
810 Retourne le nom du fichier correspondant à un numero d'unité
811 logique (entier) ainsi que le source contenu dans le fichier
814 # Si le JDC est relié à une application maitre, on délègue la recherche
815 file,text= self.appli.get_file(unite,fic_origine)
819 if os.path.exists("fort."+str(unite)):
820 file= "fort."+str(unite)
822 raise AsException("Impossible de trouver le fichier correspondant"
823 " a l unite %s" % unite)
824 if not os.path.exists(file):
825 raise AsException("%s n'est pas un fichier existant" % unite)
829 #if file == None : return None,None
830 text=string.replace(text,'\r\n','\n')
832 linecache.cache[file]=0,0,string.split(text,'\n'),file
835 def isvalid(self,cr='non'):
836 if hasattr(self,'valid'): old_valid=self.valid
838 valid=Validation.V_JDC.JDC.isvalid(self,cr)
839 if valid != old_valid:
840 CONNECTOR.Emit(self,"valid")