]> SALOME platform Git repositories - tools/eficas.git/blob - Ihm/I_ETAPE.py
Salome HOME
d902f6c0ec7c3223049ccaa3f1b6b503b1659870
[tools/eficas.git] / Ihm / I_ETAPE.py
1 #            CONFIGURATION MANAGEMENT OF EDF VERSION
2 # ======================================================================
3 # COPYRIGHT (C) 1991 - 2002  EDF R&D                  WWW.CODE-ASTER.ORG
4 # THIS PROGRAM IS FREE SOFTWARE; YOU CAN REDISTRIBUTE IT AND/OR MODIFY
5 # IT UNDER THE TERMS OF THE GNU GENERAL PUBLIC LICENSE AS PUBLISHED BY
6 # THE FREE SOFTWARE FOUNDATION; EITHER VERSION 2 OF THE LICENSE, OR
7 # (AT YOUR OPTION) ANY LATER VERSION.
8 #
9 # THIS PROGRAM IS DISTRIBUTED IN THE HOPE THAT IT WILL BE USEFUL, BUT
10 # WITHOUT ANY WARRANTY; WITHOUT EVEN THE IMPLIED WARRANTY OF
11 # MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE. SEE THE GNU
12 # GENERAL PUBLIC LICENSE FOR MORE DETAILS.
13 #
14 # YOU SHOULD HAVE RECEIVED A COPY OF THE GNU GENERAL PUBLIC LICENSE
15 # ALONG WITH THIS PROGRAM; IF NOT, WRITE TO EDF R&D CODE_ASTER,
16 #    1 AVENUE DU GENERAL DE GAULLE, 92141 CLAMART CEDEX, FRANCE.
17 #
18 #
19 # ======================================================================
20 """
21 """
22 # Modules Python
23 import string,types
24 from copy import copy
25
26 # import rajoutés suite à l'ajout de Build_sd --> à résorber
27 import traceback
28 import Noyau
29 from Noyau import N_Exception
30 from Noyau.N_Exception import AsException
31 # fin import à résorber
32
33 # Modules EFICAS
34 import I_MCCOMPO
35
36 class ETAPE(I_MCCOMPO.MCCOMPO):
37
38    def __init__(self):
39       self.niveau=self.jdc
40
41    def ident(self):
42       return self.nom
43
44    def get_sdname(self):
45       if CONTEXT.debug : print "SDNAME ",self.reuse,self.sd,self.sd.get_name()
46       if self.reuse != None:
47         sdname= self.reuse.get_name()
48       else:
49         sdname=self.sd.get_name()
50       if string.find(sdname,'sansnom') != -1 or string.find(sdname,'SD_') != -1:
51         # dans le cas où la SD est 'sansnom' ou 'SD_' on retourne la chaîne vide
52         return ''
53       return sdname
54
55    def is_reentrant(self):
56       """ 
57           Indique si la commande est reentrante
58       """
59       return self.definition.reentrant == 'o' 
60
61    def init_modif(self):
62       """
63          Met l'état de l'étape à : modifié
64          Propage la modification au parent
65          Si la fonction op_init existe, l'active
66       """
67       # Une action
68       # doit etre realisée apres init_modif et la validite reevaluée
69       # apres cette action. L'état modified de tous les objets doit etre
70       # preservé.
71       self.state = 'modified'
72       if self.parent:
73         self.parent.init_modif()
74
75    def fin_modif(self):
76       """
77           Méthode appelée une fois qu'une modification a été faite afin de 
78           déclencher d'éventuels traitements post-modification
79           ex : INCLUDE et POURSUITE
80       """
81       if self.isvalid() :
82          if type(self.definition.op_init) == types.FunctionType :
83             # XXX Normalement en mode editeur g_context ne peut pas etre utilisé
84             apply(self.definition.op_init,(self,self.parent.g_context))   
85       self.state = 'modified'
86     
87    def nomme_sd(self,nom) :
88       """
89           Cette méthode a pour fonction de donner un nom (nom) au concept 
90           produit par l'étape (self).
91           - si le concept n'existe pas, on essaye de le créer (à condition que l'étape soit valide ET non réentrante)
92           - si il existe déjà, on le renomme et on répercute les changements dans les autres étapes    
93           Les valeurs de retour sont :
94            0 si le nommage n'a pas pu etre mené à son terme,
95            1 dans le cas contraire
96       """
97       if len(nom) > 8 and self.jdc.definition.code == 'ASTER':
98         return 0,"Nom de concept trop long (maxi 8 caractères)"
99       self.init_modif()
100       if not self.isvalid(sd='non') : return 0,"Nommage du concept refusé : l'opérateur n'est pas valide"
101       #
102       # Cas particulier des opérateurs obligatoirement réentrants
103       #
104       if self.definition.reentrant == 'o':
105         # FR : appel à get_sdprod incorrect : il faut appeler get_sd_avant_etape
106         #self.sd = self.reuse = self.jdc.get_sdprod(nom)
107         self.sd = self.reuse = self.jdc.get_sd_avant_etape(nom,self)
108         if self.sd != None :
109           self.sdnom=self.sd.nom
110           return 1,"Concept existant"
111         else:
112           return 0,"Opérateur réentrant mais concept non existant"
113       #
114       # Cas particulier des opérateurs facultativement réentrants
115       #
116       old_reuse=None
117       if self.definition.reentrant == 'f' :
118         sd = self.jdc.get_sd_avant_etape(nom,self)
119         if sd != None :
120           # FR : il faut tester que la sd trouvée est du bon type !!!!!!!!!!!!!!!!!
121           if isinstance(sd,self.get_type_produit()) :
122              self.sd = self.reuse = sd
123              self.sdnom = sd.nom
124              return 1,"Opérateur facultativement réentrant et concept existant trouvé"
125           else:
126              return 0,"Concept déjà existant et de mauvais type"
127         else :
128           # il faut enlever le lien vers une SD existante car si on passe ici
129           # cela signifie que l'opérateur n'est pas utilisé en mode réentrant.
130           # Si on ne fait pas cela, on risque de modifier une SD produite par un autre opérateur
131           if self.reuse :
132              old_reuse=self.reuse
133              self.sd = self.reuse = self.sdnom = None
134       #
135       # On est dans le cas ou l'opérateur n'est pas réentrant ou est facultativement reentrant
136       # mais est utilisé en mode non réentrant
137       #
138       if self.sd == None :
139           if self.parent.get_sd_autour_etape(nom,self):
140             # Un concept de ce nom existe dans le voisinage de l'etape courante
141             # On force self.valid a 0 car l appel a isvalid precedent l a mis a 1
142             # mais ceci indique seulement une validité partielle
143             # isvalid ne devrait peut etre pas mettre l attribut valid à 1 si sd == 'non'
144             self.valid=0
145             # On retablit l'ancien concept reentrant s'il existait
146             if old_reuse:
147                self.sd=self.reuse=old_reuse
148                self.sdnom=old_reuse.nom
149             return 0,"Nommage du concept refuse : un concept de meme nom existe deja"
150           else:
151             # Il n'existe pas de concept de ce nom dans le voisinage de l'etape courante
152             # On peut donc créer le concept retourné.
153             # Il est créé sans nom mais enregistré dans la liste des concepts existants
154             self.get_sd_prod()
155             # Il suffit de changer son attribut nom pour le nommer
156             self.sd.nom = nom
157             self.sdnom=nom
158             return 1,"Nommage du concept effectué"
159       else :
160           old_nom=self.sd.nom
161           if string.find(old_nom,'sansnom') :
162             # Dans le cas où old_nom == sansnom, isvalid retourne 0 alors que ...
163             # par contre si le concept existe et qu'il s'appelle sansnom c'est que l'étape est valide
164             # on peut donc le nommer sans test préalable
165             self.sd.nom=nom
166             self.sdnom=nom
167             return 1,"Nommage du concept effectué"
168           if self.isvalid() :
169             # Normalement l appel de isvalid a mis a jour le concept produit (son type)
170             # Il suffit de spécifier l attribut nom de sd pour le nommer si le nom n est pas
171             # deja attribué
172             if self.parent.get_sd_autour_etape(nom,self):
173               return 0,"Nommage du concept refuse : un concept de meme nom existe deja"
174             else:
175               self.sd.nom=nom
176               self.sdnom=nom
177               return 1,"Nommage du concept effectué"
178           else:
179             # Normalement on ne devrait pas passer ici
180             return 0,'Normalement on ne devrait pas passer ici'
181
182    def get_sdprods(self,nom_sd):
183       """ 
184          Fonction : retourne le concept produit par l etape de nom nom_sd
185                     s il existe sinon None
186       """
187       if self.sd:
188         if self.sd.nom == nom_sd:return self.sd
189
190    def active(self):
191       """
192           Rend l'etape courante active.
193           Il faut ajouter la sd si elle existe au contexte global du JDC
194           et à la liste des sd
195       """
196       self.actif = 1
197       if not self.sd : return
198       # XXX Pourquoi faut-il faire ce qui suit ??? par defaut les etapes sont actives
199       try:
200         self.jdc.append_sdprod(self.sd)
201       except:
202         pass
203
204    def inactive(self):
205       """
206           Rend l'etape courante inactive
207           Il faut supprimer la sd du contexte global du JDC
208           et de la liste des sd
209       """
210       self.actif = 0
211       if not self.sd : return
212       self.jdc.del_sdprod(self.sd)
213       self.jdc.delete_concept_after_etape(self,self.sd)
214
215    def supprime_sdprods(self):
216       """ 
217           Fonction:
218             Lors d'une destruction d'etape, detruit tous les concepts produits
219             Un opérateur n a qu un concept produit 
220             Une procedure n'en a aucun
221             Une macro en a en général plus d'un
222       """
223       # XXX pour les macros il faudrait peut etre aussi 
224       #     supprimer les concepts a droite du = ???
225       if not self.is_reentrant() :
226         # l'étape n'est pas réentrante
227         # le concept retourné par l'étape est à supprimer car il était 
228         # créé par l'étape
229         if self.sd != None :
230           self.parent.del_sdprod(self.sd)
231           self.parent.delete_concept(self.sd)
232
233    def delete_concept(self,sd):
234       """ 
235           Inputs :
236              sd=concept detruit
237           Fonction :
238              Mettre a jour les mots cles de l etape et eventuellement 
239              le concept produit si reuse
240              suite à la disparition du concept sd
241              Seuls les mots cles simples MCSIMP font un traitement autre 
242              que de transmettre aux fils
243       """
244       if self.reuse and self.reuse == sd:
245         self.sd=self.reuse=None
246         self.init_modif()
247       for child in self.mc_liste :
248         child.delete_concept(sd)
249
250    def make_register(self):
251       """
252          Initialise les attributs jdc, id, niveau et réalise les
253          enregistrements nécessaires
254          Pour EFICAS, on tient compte des niveaux
255       """
256       if self.parent :
257          self.jdc = self.parent.get_jdc_root()
258          self.id=   self.parent.register(self)
259          if self.definition.niveau :
260             # La définition est dans un niveau. En plus on
261             # l'enregistre dans le niveau
262             self.nom_niveau_definition = self.definition.niveau.nom
263             self.niveau = self.parent.dict_niveaux[self.nom_niveau_definition]
264             self.niveau.register(self)
265          else:
266             # La définition est au niveau global
267             self.nom_niveau_definition = 'JDC'
268             self.niveau=self.parent
269       else:
270          self.jdc = self.parent =None
271          self.id=None
272          self.niveau=None
273
274    def copy(self):
275       """ Méthode qui retourne une copie de self non enregistrée auprès du JDC
276           et sans sd 
277       """
278       etape = copy(self)
279       etape.sd = None
280       etape.state = 'modified'
281       etape.reuse = None
282       etape.sdnom = None
283       etape.etape=etape
284       etape.mc_liste=[]
285       for objet in self.mc_liste:
286         new_obj = objet.copy()
287         new_obj.reparent(etape)
288         etape.mc_liste.append(new_obj)
289       return etape
290
291    def get_noms_sd_oper_reentrant(self):
292       """ 
293           Retourne la liste des noms de concepts utilisés à l'intérieur de la commande
294           qui sont du type que peut retourner cette commande 
295       """
296       liste_sd = self.get_sd_utilisees()
297       l_noms = []
298       if type(self.definition.sd_prod) == types.FunctionType:
299         d=self.cree_dict_valeurs(self.mc_liste)
300         try:
301           classe_sd_prod = apply(self.definition.sd_prod,(),d)
302         except:
303           return []
304       else:
305         classe_sd_prod = self.definition.sd_prod
306       for sd in liste_sd :
307         if sd.__class__ is classe_sd_prod : l_noms.append(sd.nom)
308       l_noms.sort()
309       return l_noms
310
311    def get_sd_utilisees(self):
312       """ 
313           Retourne la liste des concepts qui sont utilisés à l'intérieur d'une commande
314           ( comme valorisation d'un MCS) 
315       """
316       l=[]
317       for child in self.mc_liste:
318         l.extend(child.get_sd_utilisees())
319       return l
320
321    def get_genealogie(self):
322       """ 
323           Retourne la liste des noms des ascendants de l'objet self
324           en s'arretant à la première ETAPE rencontrée
325       """
326       return [self.nom]
327
328    def reparent(self,parent):
329      """
330          Cette methode sert a reinitialiser la parente de l'objet
331      """
332      self.parent=parent
333      self.jdc=parent.get_jdc_root()
334      self.etape=self
335      for mocle in self.mc_liste:
336         mocle.reparent(self)
337
338    def verif_existence_sd(self):
339      """
340         Vérifie que les structures de données utilisées dans self existent bien dans le contexte
341         avant étape, sinon enlève la référence à ces concepts
342      """
343      for motcle in self.mc_liste :
344          motcle.verif_existence_sd()
345      
346    def Build_sd(self,nom):
347       """
348          Construit le concept produit de l'opérateur. Deux cas 
349          peuvent se présenter :
350         
351          - le parent n'est pas défini. Dans ce cas, l'étape prend en charge la création 
352            et le nommage du concept.
353
354          - le parent est défini. Dans ce cas, l'étape demande au parent la création et 
355            le nommage du concept.
356
357       """
358       if not self.isactif():return
359       # FR : attention cette méthode ne devrait pas se trouver là car elle surcharge celle qui 
360       # se trouve dans N_ETAPE.py et elle est partie intégrante du noyau, mais, suite à l'absence de 
361       # test de validité de l'opérateur avant d'essayer de déterminer la sd produite, on n'arrivait
362       # pas à relire avec EFICAS un fichier contenant une étape encore incomplète du style :
363       #  sansnom = AFFE_CHAR_CINE(MODELE=None)
364       # Suite à la stabilisation du noyau d'Aster, je n'ai pas eu d'autre solution que de surcharger
365       # cette méthode ici en rajoutant le test manquant ...
366       # CCAR : cette modification ne corrige le probleme qu'en partie. Il faudrait probablement
367       # supprimer les erreurs fatales (exception ) et retourner systematiquement un objet produit
368       # meme en cas d'erreur et reporter l'emission du message d'erreur a la phase de validation
369       if not self.isvalid(sd='non') : return
370       else:self.state='undetermined'
371       self.sdnom=nom
372       try:
373          if self.parent:
374             sd= self.parent.create_sdprod(self,nom)
375             if type(self.definition.op_init) == types.FunctionType: 
376                apply(self.definition.op_init,(self,self.parent.g_context))
377          else:
378             sd=self.get_sd_prod()
379             # On n'utilise pas self.definition.op_init car self.parent 
380             # n'existe pas
381             if sd != None and self.reuse == None:
382                # On ne nomme le concept que dans le cas de non reutilisation 
383                # d un concept
384                sd.nom=nom
385          if self.jdc and self.jdc.par_lot == "NON" :
386             self.Execute()
387          return sd
388       except AsException,e:
389          raise AsException("Etape ",self.nom,'ligne : ',self.appel[0],
390                               'fichier : ',self.appel[1],e)
391       except EOFError:
392          # XXX Normalement le contexte courant doit etre le parent.
393          # Il n'y a pas de raison de remettre le contexte au parent
394          #self.reset_current_step()
395          raise
396       except :
397          l=traceback.format_exception(sys.exc_info()[0],sys.exc_info()[1],sys.exc_info()[2])
398          raise AsException("Etape ",self.nom,'ligne : ',self.appel[0],
399                            'fichier : ',self.appel[1]+'\n',
400                             string.join(l))
401      
402      
403      
404      
405      
406      
407      
408      
409      
410      
411      
412      
413      
414      
415      
416      
417      
418      
419      
420         
421