X-Git-Url: http://git.onelab.eu/?a=blobdiff_plain;f=sfa%2Fstorage%2Fmodel.py;h=3be257ffd0fe3d5a1374f4dfaaa040420ade31e6;hb=7fcb8153bd74ce00f9f0d43c8aa090e0b70e345f;hp=780853ccbf3dba44fc9c5b8875e4b60b17618cfb;hpb=23f547bfc848537631535b1677d1409f18ef91cd;p=sfa.git diff --git a/sfa/storage/model.py b/sfa/storage/model.py index 780853cc..3be257ff 100644 --- a/sfa/storage/model.py +++ b/sfa/storage/model.py @@ -114,6 +114,10 @@ class RegRecord (Base,AlchemyObj): result += ">" return result + # shortcut - former implem. was record-based + def get (self, field, default): + return getattr(self,field,default) + @validates ('gid') def validate_gid (self, key, gid): if gid is None: return @@ -123,6 +127,8 @@ class RegRecord (Base,AlchemyObj): def validate_datetime (self, key, incoming): if isinstance (incoming, datetime): return incoming elif isinstance (incoming, (int,float)):return datetime.fromtimestamp (incoming) + else: logger.info("Cannot validate datetime for key %s with input %s"%\ + (key,incoming)) @validates ('date_created') def validate_date_created (self, key, incoming): return self.validate_datetime (key, incoming) @@ -144,6 +150,20 @@ class RegRecord (Base,AlchemyObj): now=datetime.now() self.last_updated=now +#################### cross-relations tables +# authority x user (pis) association +authority_pi_table = \ + Table ( 'authority_pi', Base.metadata, + Column ('authority_id', Integer, ForeignKey ('records.record_id'), primary_key=True), + Column ('pi_id', Integer, ForeignKey ('records.record_id'), primary_key=True), + ) +# slice x user (researchers) association +slice_researcher_table = \ + Table ( 'slice_researcher', Base.metadata, + Column ('slice_id', Integer, ForeignKey ('records.record_id'), primary_key=True), + Column ('researcher_id', Integer, ForeignKey ('records.record_id'), primary_key=True), + ) + ############################## # all subclasses define a convenience constructor with a default value for type, # and when applicable a way to define local fields in a kwd=value argument @@ -152,6 +172,13 @@ class RegAuthority (RegRecord): __tablename__ = 'authorities' __mapper_args__ = { 'polymorphic_identity' : 'authority' } record_id = Column (Integer, ForeignKey ("records.record_id"), primary_key=True) + #### extensions come here + reg_pis = relationship \ + ('RegUser', + secondary=authority_pi_table, + primaryjoin=RegRecord.record_id==authority_pi_table.c.authority_id, + secondaryjoin=RegRecord.record_id==authority_pi_table.c.pi_id, + backref='reg_authorities_as_pi') def __init__ (self, **kwds): # fill in type if not previously set @@ -163,13 +190,15 @@ class RegAuthority (RegRecord): def __repr__ (self): return RegRecord.__repr__(self).replace("Record","Authority") -#################### -# slice x user (researchers) association -slice_researcher_table = \ - Table ( 'slice_researcher', Base.metadata, - Column ('slice_id', Integer, ForeignKey ('records.record_id'), primary_key=True), - Column ('researcher_id', Integer, ForeignKey ('records.record_id'), primary_key=True), - ) + def update_pis (self, pi_hrns): + # don't ruin the import of that file in a client world + from sfa.storage.alchemy import dbsession + # strip that in case we have words + pi_hrns = [ x.strip() for x in pi_hrns ] + request = dbsession.query (RegUser).filter(RegUser.hrn.in_(pi_hrns)) + logger.info ("RegAuthority.update_pis: %d incoming pis, %d matches found"%(len(pi_hrns),request.count())) + pis = dbsession.query (RegUser).filter(RegUser.hrn.in_(pi_hrns)).all() + self.reg_pis = pis #################### class RegSlice (RegRecord): @@ -182,7 +211,7 @@ class RegSlice (RegRecord): secondary=slice_researcher_table, primaryjoin=RegRecord.record_id==slice_researcher_table.c.slice_id, secondaryjoin=RegRecord.record_id==slice_researcher_table.c.researcher_id, - backref="reg_slices_as_researcher") + backref='reg_slices_as_researcher') def __init__ (self, **kwds): if 'type' not in kwds: kwds['type']='slice' @@ -191,6 +220,28 @@ class RegSlice (RegRecord): def __repr__ (self): return RegRecord.__repr__(self).replace("Record","Slice") + def update_researchers (self, researcher_hrns): + # don't ruin the import of that file in a client world + from sfa.storage.alchemy import dbsession + # strip that in case we have words + researcher_hrns = [ x.strip() for x in researcher_hrns ] + request = dbsession.query (RegUser).filter(RegUser.hrn.in_(researcher_hrns)) + logger.info ("RegSlice.update_researchers: %d incoming researchers, %d matches found"%(len(researcher_hrns),request.count())) + researchers = dbsession.query (RegUser).filter(RegUser.hrn.in_(researcher_hrns)).all() + self.reg_researchers = researchers + + # when dealing with credentials, we need to retrieve the PIs attached to a slice + def get_pis (self): + # don't ruin the import of that file in a client world + from sfa.storage.alchemy import dbsession + from sfa.util.xrn import get_authority + authority_hrn = get_authority(self.hrn) + auth_record = dbsession.query(RegAuthority).filter_by(hrn=authority_hrn).first() + return auth_record.reg_pis + + @validates ('expires') + def validate_expires (self, key, incoming): return self.validate_datetime (key, incoming) + #################### class RegNode (RegRecord): __tablename__ = 'nodes' @@ -310,3 +361,35 @@ def make_record_xml (xml): logger.info("load from xml, keys=%s"%xml_dict.keys()) return make_record_dict (xml_dict) +#################### +# augment local records with data from builtin relationships +# expose related objects as a list of hrns +# we pick names that clearly won't conflict with the ones used in the old approach, +# were the relationships data came from the testbed side +# for each type, a dict of the form {:} +# so after that, an 'authority' record will e.g. have a 'reg-pis' field with the hrns of its pi-users +augment_map={'authority': {'reg-pis':'reg_pis',}, + 'slice': {'reg-researchers':'reg_researchers',}, + 'user': {'reg-pi-authorities':'reg_authorities_as_pi', + 'reg-slices':'reg_slices_as_researcher',}, + } + +def augment_with_sfa_builtins (local_record): + # don't ruin the import of that file in a client world + from sfa.util.xrn import Xrn + # add a 'urn' field + setattr(local_record,'reg-urn',Xrn(xrn=local_record.hrn,type=local_record.type).urn) + # users have keys and this is needed to synthesize 'users' sent over to CreateSliver + if local_record.type=='user': + user_keys = [ key.key for key in local_record.reg_keys ] + setattr(local_record, 'reg-keys', user_keys) + # search in map according to record type + type_map=augment_map.get(local_record.type,{}) + # use type-dep. map to do the job + for (field_name,attribute) in type_map.items(): + # get related objects + related_records = getattr(local_record,attribute,[]) + hrns = [ r.hrn for r in related_records ] + setattr (local_record, field_name, hrns) + +