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.
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.
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.
19 # ======================================================================
25 myrepr.maxstring = 100
28 from Noyau.N_utils import repr_float
30 # Attention : les classes ASSD,.... peuvent etre surchargées
31 # dans le package Accas. Il faut donc prendre des précautions si
32 # on utilise les classes du Noyau pour faire des tests (isxxxx, ...)
33 # Si on veut créer des objets comme des CO avec les classes du noyau
34 # ils n'auront pas les conportements des autres packages (pb!!!)
35 # Il vaut mieux les importer d'Accas mais problème d'import circulaire,
36 # on ne peut pas les importer au début.
37 # On fait donc un import local quand c'est nécessaire (peut occasionner
38 # des pbs de prformance).
39 from Noyau.N_ASSD import ASSD,assd
40 from Noyau.N_GEOM import GEOM,geom
41 from Noyau.N_CO import CO
44 from Extensions import parametre
47 class MCSIMP(I_OBJECT.OBJECT):
50 Retourne le texte à afficher dans l'arbre représentant la valeur de l'objet
53 if self.valeur == None :
55 elif type(self.valeur) == types.FloatType :
56 #txt = repr_float(self.valeur)
57 # Normalement str fait un travail correct
58 txt = str(self.valeur)
59 elif type(self.valeur) in (types.ListType,types.TupleType) :
62 for val in self.valeur:
63 if type(val) == types.FloatType :
64 # Normalement str fait un travail correct
65 #txt=txt + i*',' + repr_float(val)
66 txt=txt + i*',' + str(val)
67 elif type(val) == types.InstanceType and isinstance(val,ASSD):
68 txt = txt + i*',' + val.get_name()
71 elif val.__class__.__name__ in ('PARAMETRE','PARAMETRE_EVAL'):
72 txt = txt + i*','+ str(val)
74 txt = txt + i*','+ myrepr.repr(val)
79 if type(txt) != types.StringType:
80 if type(txt) == types.InstanceType:
81 if isinstance(txt,parametre.PARAMETRE):
84 # txt peut etre une longue chaine sur plusieurs lignes.
85 # Il est possible de tronquer cette chaine au premier \n et
86 # de limiter la longueur de la chaine a 30 caracteres. Cependant
87 # ceci provoque une perte d'information pour l'utilisateur
88 # Pour le moment on retourne la chaine telle que
91 # Partie de code inaccessible (pour memoire)
92 # txt est tronquee au dela d'un certain nombre de caractères
93 # et avant tout retour chariot (txt peut etre une chaine de caractères
94 # sur plusieurs lignes (ex:shell)
95 txt = string.split(txt,'\n')[0]
103 Retourne une chaîne de caractère représentant la valeur de self
106 if type(val) != types.TupleType :
108 return val.get_name()
115 s=s+item.get_name()+','
123 Méthode booléenne qui retourne 1 si l'objet attend un objet ASSD
124 qui n'existe pas encore (type CO()), 0 sinon
126 for typ in self.definition.type:
127 if type(typ) == types.ClassType :
128 if issubclass(typ,CO) :
134 Méthode booléenne qui retourne 1 si le MCS attend un objet de type ASSD
137 for typ in self.definition.type:
138 if type(typ) == types.ClassType :
139 if issubclass(typ,ASSD) and not issubclass(typ,GEOM):
143 def wait_assd_or_geom(self):
145 Retourne 1 si le mot-clé simple attend un objet de type
146 assd, ASSD, geom ou GEOM
147 Retourne 0 dans le cas contraire
149 for typ in self.definition.type:
150 if type(typ) == types.ClassType :
151 if typ.__name__ in ("GEOM","ASSD","geom","assd") or issubclass(typ,GEOM) :
157 Retourne 1 si le mot-clé simple attend un objet de type GEOM
158 Retourne 0 dans le cas contraire
160 for typ in self.definition.type:
161 if type(typ) == types.ClassType :
162 if issubclass(typ,GEOM) : return 1
167 Retourne 1 si le mot-clé simple attend un objet de type TXM
168 Retourne 0 dans le cas contraire
170 for typ in self.definition.type:
171 if typ == 'TXM' :return 1
174 def get_liste_valeurs(self):
177 if self.valeur == None:
179 elif type(self.valeur) == types.TupleType:
180 return list(self.valeur)
181 elif type(self.valeur) == types.ListType:
187 return self.definition.statut=='o'
189 # def set_valeur(self,new_valeur,evaluation='oui'):
191 # Remplace la valeur de self(si elle existe) par new_valeur
192 # - si evaluation = 'oui' :
193 # essaie d'évaluer new_valeur dans le contexte
194 # - si evaluation = 'non' :
195 # n'essaie pas d'évaluer (on stocke une string ou
196 # une valeur de la liste into )
198 # if evaluation == 'oui' and not self.wait_assd_or_geom():
199 # valeur,test = self.eval_valeur(new_valeur)
201 # self.val = new_valeur
202 # self.valeur = valeur
207 # # On n'a pas trouve de concept ni réussi à évaluer la valeur
209 # # Si le mot cle simple attend un type CO on crée un objet de ce
210 # # type de nom new_valeur
213 # # Pour avoir la classe CO avec tous ses comportements
214 # from Accas import CO
215 # self.valeur=CO(new_valeur)
217 # traceback.print_exc()
220 # self.val=self.valeur
223 # elif type(new_valeur)==types.StringType and self.wait_TXM():
225 # self.val = new_valeur
226 # self.valeur = new_valeur
232 # on ne fait aucune vérification ...
233 def set_valeur(self,new_valeur,evaluation='oui'):
235 self.valeur = new_valeur
236 self.val = new_valeur
240 def eval_valeur(self,new_valeur):
242 Essaie d'évaluer new_valeur comme une SD, une déclaration Python
243 ou un EVAL: Retourne la valeur évaluée (ou None) et le test de réussite (1 ou 0)
245 sd = self.jdc.get_contexte_avant(self.etape).get(new_valeur,None)
250 # On veut EVAL avec tous ses comportements. On utilise Accas. Perfs ??
251 from Accas import EVAL
254 objet = eval(new_valeur,d)
257 if CONTEXT.debug : traceback.print_exc()
260 def delete_concept(self,sd):
265 Met a jour la valeur du mot cle simple suite à la disparition
268 if type(self.valeur) == types.TupleType :
269 if sd in self.valeur:
270 self.valeur=list(self.valeur)
271 self.valeur.remove(sd)
273 elif type(self.valeur) == types.ListType:
274 if sd in self.valeur:
275 self.valeur.remove(sd)
278 if self.valeur == sd:
283 def replace_concept(self,old_sd,sd):
286 - old_sd=concept remplacé
289 Met a jour la valeur du mot cle simple suite au remplacement
292 if type(self.valeur) == types.TupleType :
293 if old_sd in self.valeur:
294 self.valeur=list(self.valeur)
295 i=self.valeur.index(old_sd)
298 elif type(self.valeur) == types.ListType:
299 if old_sd in self.valeur:
300 i=self.valeur.index(old_sd)
304 if self.valeur == old_sd:
310 """ Retourne une copie de self """
311 objet = self.makeobjet()
312 # il faut copier les listes et les tuples mais pas les autres valeurs
313 # possibles (réel,SD,...)
314 if type(self.valeur) in (types.ListType,types.TupleType):
315 objet.valeur = copy(self.valeur)
317 objet.valeur = self.valeur
318 objet.val = objet.valeur
322 return self.definition(val = None, nom = self.nom,parent = self.parent)
324 def get_sd_utilisees(self):
326 Retourne une liste qui contient la SD utilisée par self si c'est le cas
327 ou alors une liste vide
330 if type(self.valeur) == types.InstanceType:
331 #XXX Est ce différent de isinstance(self.valeur,ASSD) ??
332 if issubclass(self.valeur.__class__,ASSD) : l.append(self.valeur)
336 def set_valeur_co(self,nom_co):
338 Affecte à self l'objet de type CO et de nom nom_co
340 step=self.etape.parent
341 if nom_co == None or nom_co == '':
344 # Pour le moment on importe en local le CO de Accas.
345 # Si problème de perfs, il faudra faire autrement
347 # Avant de créer un concept il faut s'assurer du contexte : step
349 sd= step.get_sd_autour_etape(nom_co,self.etape,avec='oui')
351 # Si un concept du meme nom existe deja dans la portée de l'étape
352 # on ne crée pas le concept
353 return 0,"un concept de meme nom existe deja"
354 # Il n'existe pas de concept de meme nom. On peut donc le créer
355 # Il faut néanmoins que la méthode NommerSdProd de step gère les
356 # contextes en mode editeur
357 # Normalement la méthode du Noyau doit etre surchargée
358 # On déclare l'étape du mot clé comme etape courante pour NommerSdprod
359 cs= CONTEXT.get_current_step()
360 CONTEXT.unset_current_step()
361 CONTEXT.set_current_step(step)
362 step.set_etape_context(self.etape)
363 new_objet = CO(nom_co)
364 CONTEXT.unset_current_step()
365 CONTEXT.set_current_step(cs)
367 self.valeur = new_objet
371 # On force l'enregistrement de new_objet en tant que concept produit
372 # de la macro en appelant get_type_produit avec force=1
373 self.etape.get_type_produit(force=1)
374 return 1,"Concept créé"
376 def reparent(self,parent):
378 Cette methode sert a reinitialiser la parente de l'objet
382 self.etape=parent.etape
384 def verif_existence_sd(self):
386 Vérifie que les structures de données utilisées dans self existent bien dans le contexte
387 avant étape, sinon enlève la référence à ces concepts
389 l_sd_avant_etape = self.jdc.get_contexte_avant(self.etape).values()
390 if type(self.valeur) in (types.TupleType,types.ListType) :
392 for sd in self.valeur:
393 if isinstance(sd,ASSD) :
394 if sd in l_sd_avant_etape :
399 # Est ce init_modif ou init_modif_up
400 # Normalement init_modif va avec fin_modif
404 if isinstance(self.valeur,ASSD) :
405 if self.valeur not in l_sd_avant_etape :
410 def get_min_max(self):
412 Retourne les valeurs min et max admissibles pour la valeur de self
414 return self.definition.min,self.definition.max
419 Retourne le type attendu par le mot-clé simple
421 return self.definition.type
423 #ATTENTION SURCHARGE : toutes les methodes ci apres sont des surcharges du Noyau et de Validation
424 # Elles doivent etre reintegrees des que possible
426 def is_complexe(self,valeur):
427 """ Retourne 1 si valeur est un complexe, 0 sinon """
428 if type(valeur) == types.InstanceType :
429 #XXX je n'y touche pas pour ne pas tout casser mais il serait
430 #XXX préférable d'appeler une méthode de valeur : return valeur.is_type('C'), par exemple
431 if valeur.__class__.__name__ in ('EVAL','complexe','PARAMETRE_EVAL'):
433 elif valeur.__class__.__name__ in ('PARAMETRE',):
434 # il faut tester si la valeur du parametre est un entier
435 #XXX ne serait ce pas plutot complexe ???? sinon expliquer
436 return self.is_entier(valeur.valeur)
438 print "Objet non reconnu dans is_complexe %s" %`valeur`
440 # Pour permettre l'utilisation de complexes Python
441 #elif type(valeur) == types.ComplexType:
443 elif type(valeur) != types.TupleType :
446 if len(valeur) != 3 :
449 if type(valeur[0]) != types.StringType : return 0
450 if string.strip(valeur[0]) not in ('RI','MP'):
453 if not self.is_reel(valeur[1]) or not self.is_reel(valeur[2]) : return 0
456 def is_reel(self,valeur):
458 Retourne 1 si valeur est un reel, 0 sinon
460 if type(valeur) == types.InstanceType :
461 #XXX je n'y touche pas pour ne pas tout casser mais il serait
462 #XXX préférable d'appeler une méthode de valeur : return valeur.is_type('R'), par exemple
463 #XXX ou valeur.is_reel()
464 #XXX ou encore valeur.compare(self.is_reel)
465 if valeur.__class__.__name__ in ('EVAL','reel','PARAMETRE_EVAL') :
467 elif valeur.__class__.__name__ in ('PARAMETRE',):
468 # il faut tester si la valeur du parametre est un réel
469 return self.is_reel(valeur.valeur)
471 print "Objet non reconnu dans is_reel %s" %`valeur`
473 elif type(valeur) not in (types.IntType,types.FloatType,types.LongType):
474 # ce n'est pas un réel
479 def is_entier(self,valeur):
480 """ Retourne 1 si valeur est un entier, 0 sinon """
481 if type(valeur) == types.InstanceType :
482 #XXX je n'y touche pas pour ne pas tout casser mais il serait
483 #XXX préférable d'appeler une méthode de valeur : return valeur.is_type('I'), par exemple
484 if valeur.__class__.__name__ in ('EVAL','entier','PARAMETRE_EVAL') :
486 elif valeur.__class__.__name__ in ('PARAMETRE',):
487 # il faut tester si la valeur du parametre est un entier
488 return self.is_entier(valeur.valeur)
490 print "Objet non reconnu dans is_reel %s" %`valeur`
492 elif type(valeur) not in (types.IntType,types.LongType):
493 # ce n'est pas un entier
498 def is_object_from(self,objet,classe):
500 Retourne 1 si valeur est un objet de la classe classe ou d'une
501 sous-classe de classe, 0 sinon
503 if type(objet) != types.InstanceType :
505 if not objet.__class__ == classe and not issubclass(objet.__class__,classe):
511 if hasattr(self,'valid'):
517 def set_valid(self,valid):
518 old_valid=self.get_valid()
520 self.state = 'unchanged'
521 if not old_valid or old_valid != self.valid :
524 def isvalid(self,cr='non'):
526 Cette méthode retourne un indicateur de validité de l'objet de type MCSIMP
528 - 0 si l'objet est invalide
529 - 1 si l'objet est valide
531 Le paramètre cr permet de paramétrer le traitement. Si cr == 'oui'
532 la méthode construit également un comte-rendu de validation
533 dans self.cr qui doit avoir été créé préalablement.
535 if self.state == 'unchanged':
540 # verifiaction presence
541 if self.isoblig() and v == None :
543 self.cr.fatal(string.join(("Mot-clé : ",self.nom," obligatoire non valorisé")))
549 self.cr.fatal("None n'est pas une valeur autorisée")
552 valid = self.verif_type(val=v,cr=cr)*self.verif_into(cr=cr)*self.verif_card(cr=cr)
554 # On verifie les validateurs s'il y en a et si necessaire (valid == 1)
556 if valid and self.definition.validators and not self.definition.validators.verif(self.valeur):
558 self.cr.fatal(string.join(("Mot-clé : ",self.nom,"devrait avoir ",self.definition.validators.info())))
560 # fin des validateurs
563 self.set_valid(valid)