Annotation of OSA_system2/OSAS_browser.py, revision 1.10

1.1       dwinter     1: """ Classes for displaying, browsing and organizing the archive
                      2: """
                      3: 
                      4: 
                      5: import OSAS_helpers
                      6: from AccessControl import ClassSecurityInfo
                      7: from Products.PageTemplates.PageTemplateFile import PageTemplateFile
                      8: from OFS.Folder import Folder
                      9: from OFS.SimpleItem import SimpleItem
                     10: from Globals import InitializeClass,package_home
                     11: import zLOG
                     12: import os
                     13: import os.path
                     14: import stat
1.2       dwinter    15: import xml.dom.minidom
1.3       dwinter    16: from types import *
1.8       dwinter    17: import xmlrpclib
1.10    ! dwinter    18: from OSAS_helpers import *
1.8       dwinter    19: 
1.1       dwinter    20: 
                     21: class OSAS_storeOnline(SimpleItem):
                     22:     """Webfrontend für das Storagesystem
                     23:     liefert Browserumgebung 
                     24:     """
                     25:     meta_type="OSAS_StoreOnline__neu"
                     26:     
                     27:     security=ClassSecurityInfo()
                     28: 
                     29:     _v_fileSystem={} #chache fuer filesystem
1.10    ! dwinter    30:     _v_metaFiles={} #chache fuer indexMeta
        !            31:     def getMetaFile(self,path):
        !            32:         """get index.meta and translate it to HTML"""
        !            33:         """Lies Metafile ein
        !            34:         @param path: Pfad des index.met        
        !            35:         @return: index.meta file
        !            36:         """
        !            37:         html=[]
        !            38:         server=xmlrpclib.Server(self.serverUrl)
        !            39:         
        !            40: 
        !            41:         f=server.getFile(path+"/index.meta")
        !            42:         
        !            43:         if not f:
        !            44:               
        !            45:               return self.getMetaInfoFromIndexMeta(path)
        !            46:               #return "NO_METADATA"
        !            47:         else:
        !            48:                       
        !            49:            dom = xml.dom.minidom.parseString(f)
        !            50:            
        !            51:            try:
        !            52:                name=getText(dom.getElementsByTagName("name")[0].childNodes)
        !            53:            except:
        !            54:                name="NOT_DEFINED!!!"
        !            55:            try:
        !            56:                creator=getText(dom.getElementsByTagName("creator")[0].childNodes)
        !            57:            except:
        !            58:                creator="NOT_DEFINED!!!"
        !            59: 
        !            60:            try:
        !            61:                creation_date=getText(dom.getElementsByTagName("archive-creation-date")[0].childNodes)
        !            62:            except:
        !            63:                creation_date="NOT_DEFINED!!!"
        !            64: 
        !            65:            try:
        !            66:                description=getText(dom.getElementsByTagName("description")[0].childNodes)
        !            67:            except:
        !            68:                description="NOT_DEFINED!!!"
        !            69: 
        !            70:            try:
        !            71:             type=getText(dom.getElementsByTagName("content-type")[0].childNodes) 
        !            72:            except:
        !            73:             type=""
        !            74:            if type=="scanned document":
        !            75:                     html="<h3>Document: "+name+"</h3>"
        !            76:            elif type=="folder":
        !            77:                     html="<h3>Folder: "+name+"</h3>"
        !            78:            else:
        !            79:                     html="<h3>Document: "+name+"</h3>"
        !            80: 
        !            81:            html=html+"<p><i>created by: "+creator+" at: "+creation_date+"</i></p>" 
        !            82:            html=html+"<h4>Description</h4><p>"+description+"</p>"
        !            83:            try:
        !            84:             bib = dom.getElementsByTagName("meta")[0].getElementsByTagName("bib")[0]
        !            85:             if bib.attributes.has_key('type'):
        !            86:               html=html+"<h4>Info ("+bib.attributes['type'].value+")</h4>"
        !            87:             else:
        !            88:               html=html+"<h4>Info</h4>"
        !            89:             html=html+getBib(bib.childNodes)
        !            90: 
        !            91:            except:
        !            92:             """none"""
1.8       dwinter    93: 
1.10    ! dwinter    94:     #        html=html.encode('utf-8','replace')+getBib(bib.childNodes).encode('utf-8','replace')
        !            95: 
        !            96:            return html
        !            97: 
        !            98: 
        !            99:     def getMetaInfoFromIndexMeta(self,path):
        !           100:         """metadaten zu path als html aus dem index.meta file zu path (meta tag im file bzw. dir container)
        !           101:         @param path: Pfad auf das Object relativ zum rootFolderName
        !           102:         @return: metadata als html
        !           103:         """
        !           104:         xmlInfos=self.findEntryInIndexMeta(path)
        !           105:         if xmlInfos:
        !           106:             return OSAS_helpers.getMetaInfoFromXML(path,xmlInfos)
        !           107:         else:
        !           108:             return ""
1.8       dwinter   109:    
1.10    ! dwinter   110:     def findEntryInIndexMeta(self,path):
        !           111:         """Finde im naechstgelegenden index.meta relativ zu path den entprechenden Eintrag fuer diesen Pfad.
        !           112:         @param path: Pfad auf das Object relativ zum rootFolderName
        !           113:         @return: den Teil von Index.meta der Informationen zu path enthaelt, None wenn error.
        !           114:         """
        !           115:         
        !           116:         server=xmlrpclib.Server(self.serverUrl)
        !           117:         indexMeta=server.findIndexMeta(path) # suche index.meta
        !           118:         
        !           119:         if not indexMeta:
        !           120:             return None
        !           121: 
        !           122:         realPath=os.path.split(indexMeta)[0]
        !           123:         path=os.path.normpath(path)
        !           124: 
        !           125:         try:
        !           126:             
        !           127:             dom=xml.dom.minidom.parseString(server.getFile(indexMeta))
        !           128:         except:
        !           129:             zLOG.LOG("OSAS_browser (findEntryInIndexMeta)",zLOG.ERROR,"Cannot parse: %s"%indexMeta)
        !           130:         #ist path ein directory? 
        !           131:         dirs=dom.getElementsByTagName('dir')
        !           132:         for dir in dirs:
        !           133:             pathes=dir.getElementsByTagName('path')
        !           134:             if pathes:
        !           135:                 pathX=OSAS_helpers.getText(pathes[0].childNodes)
        !           136:             else:
        !           137:                 pathX=""
        !           138:             names=dir.getElementsByTagName('name')
        !           139:             if names:
        !           140:                 name=OSAS_helpers.getText(names[0].childNodes)
        !           141:             else:
        !           142:                 name=""
        !           143: 
        !           144:             checkpath=os.path.normpath(os.path.join(realPath,pathX,name))
        !           145:             if checkpath==path:
        !           146:                 
        !           147:                 return dir.toxml()
        !           148: 
        !           149:         #ist path ein file?
        !           150:         files=dom.getElementsByTagName('file')
        !           151:         for dir in dirs:
        !           152:             pathes=dir.getElementsByTagName('path')
        !           153:             if pathes:
        !           154:                 pathX=OSAS_helpers.getText(pathes[0].childNodes)
        !           155:             else:
        !           156:                 pathX=""
        !           157:             names=dir.getElementsByTagName('name')
        !           158:             if names:
        !           159:                 name=OSAS_helpers.getText(names[0].childNodes)
        !           160:             else:
        !           161:                 name=""
        !           162: 
        !           163:             checkpath=os.path.normpath(os.path.join(realPath,pathX,name))
        !           164:             if checkpath==path:
        !           165:                 
        !           166:                 return dir.toxml()
        !           167: 
        !           168:         
        !           169:         return None
        !           170: 
        !           171: 
        !           172:     def getSubDirsFromIndexMeta(self,path):
        !           173:         
        !           174:         """Gebe alle path untergeordenten Objekte aus
        !           175:         @param path: optional, default ist "", Pfad auf das Object relativ zum rootFolderName
        !           176:         @return: Directory [pfad auf das Objekt]->(fileType,''), fileType ist hierbei OSAS_dir_archive falls Object ein directory und OSAS_file_archive falls das Object ein File ist,der zweite Eintrag des Tupels ist zur Zeit immer '', spaeter wird hier die Beschreibung gemaess Metadaten stehen, wie bei readObjectsFromPath.
        !           177:         @todo: Rueckgabe einer Beschreibung gemaess Metadaten
        !           178:         """
        !           179:         ret={}
        !           180:         server=xmlrpclib.Server(self.serverUrl)
        !           181:         indexMeta,stats=server.findIndexMetaWithStats(path)#findex index.meta zu path.
        !           182: 
        !           183:         if not indexMeta:
        !           184:             return ret
        !           185: 
        !           186:         realPath=os.path.split(indexMeta)[0]
        !           187:         path=path.replace(realPath,"")
        !           188:         if path and (path[0]==os.sep): #falls am Anfang os.sep steht lösche dieses.
        !           189:             path=path[1:]
        !           190: 
        !           191: 
        !           192: 
        !           193:         #teste ob schon im cache
        !           194:         
        !           195:         if self._v_metaFiles.has_key(path) and (self._v_metaFiles[path][0]==stats[stat.ST_MTIME]):
        !           196:             return self._v_metaFiles[path][1]
        !           197: 
        !           198:         try:
        !           199:             dom=xml.dom.minidom.parseString(server.getFile(indexMeta))
        !           200:         except:
        !           201:             zLOG.LOG("OSAS_browser (getSubDirsFromIndexMeta)",zLOG.ERROR,"Cannot parse: %s"%indexMeta)
        !           202:             return ret
        !           203: 
        !           204:         dirs=[]
        !           205:         dirs=dom.getElementsByTagName('dir')+dom.getElementsByTagName('file')
        !           206:     
        !           207:         for dir in dirs:
        !           208:             pathes=dir.getElementsByTagName('path')
        !           209:             if pathes:
        !           210:                 pathX=OSAS_helpers.getText(pathes[0].childNodes)
        !           211:             else:
        !           212:                 pathX=""
        !           213:             names=dir.getElementsByTagName('name')
        !           214:             if names:
        !           215:                 name=OSAS_helpers.getText(names[0].childNodes)
        !           216:             else:
        !           217:                 name=""
        !           218: 
        !           219:             #print "PP",pathX,path
        !           220:             if pathX==path:
        !           221:                 if dir.tagName=="dir":
        !           222:                     fileType="OSAS_dir_archive"
        !           223:                 else:
        !           224:                     fileType="OSAS_file_archive"
        !           225: 
        !           226:                 object=os.path.join(realPath,pathX,name)
        !           227:                 ret[object.encode('utf-8')]=(fileType,'')
        !           228: 
        !           229:         self._v_metaFiles[path]=(stats[stat.ST_MTIME],ret) # speicher im chache
        !           230:       
        !           231:         return ret
        !           232: 
        !           233:         
1.1       dwinter   234:     
1.8       dwinter   235:     def __init__(self,id,serverUrl):
1.7       dwinter   236:         """initialize a new instance
                    237:         @param id: Zope id"""
1.1       dwinter   238:         self.id = id
1.8       dwinter   239:         self.serverUrl = serverUrl
1.1       dwinter   240:     
                    241: 
                    242:     security.declareProtected('View','index_html')
                    243:     def index_html(self):
1.2       dwinter   244:         """main view either standard template zpt/storeOnline_index_html.zpt or storeOnline_index.html in tree"""
1.1       dwinter   245:         if hasattr(self,'storeOnline_index.html'):
                    246:             return getattr(self,'storeOnline_index.html')()
                    247:         else:
                    248:             pt=PageTemplateFile(os.path.join(package_home(globals()),'zpt','storeOnline_index_html.zpt')).__of__(self)
                    249:             return pt()
                    250: 
                    251: 
1.2       dwinter   252:     def findIndexMeta(self,path=""):
1.7       dwinter   253:         """finde Rueckwaerts im Baum von Pfad ausgehend, dass erste index.meta file
                    254:         @keyword path: default ist "", Pfad auf das Object relativ zum rootFolderName
1.2       dwinter   255:         @return: None falls kein index.meta existiert sonst Pfad auf das index.meta
                    256:         """
                    257:         realPath=os.path.normpath(os.path.join(self.rootFolderName,path))
                    258:         #suche index.meta
1.8       dwinter   259:         server=xmlrpclib.Server(self.serverUrl)
                    260:         return server.findIndexMeta(realPath)
1.7       dwinter   261:         
1.2       dwinter   262: 
1.8       dwinter   263:   
1.2       dwinter   264: 
1.9       dwinter   265:    
1.1       dwinter   266: 
1.7       dwinter   267:     def readObjectsFromPath(self,path="",metaDataId=None):
1.1       dwinter   268:         """Liest files aus dem path und speichert im cache _v_filesystem.
                    269: 
1.7       dwinter   270:         @keyword path : path relativ zum root folder des Storagesystems
                    271:         @keyword metaDataId: Optional, id des OSAS_Metadata Object, dass benutzt werden soll, generisch wird das erste Object, dass in parent gefunden wird angezeigt.
1.2       dwinter   272:         @return: directory der Form [pfad zum Objekt] -> (fileType,metadatum als String)
1.1       dwinter   273:         """
1.8       dwinter   274:         server=xmlrpclib.Server(self.serverUrl)             
1.1       dwinter   275:         realPath=os.path.normpath(os.path.join(self.rootFolderName,path))
1.7       dwinter   276: 
                    277:         if metaDataId:
                    278:             metaData=getattr(self,metaDataId)
                    279:             if not (getattr(metaData,'meta_type','')=='OSAS_Metadata__neu'):
                    280:                 zLOG.LOG('OSAS_browser (readObjectsFromPath)',zLOG.ERROR,"%s is not OSAS_Metadata")
                    281:                 metaData=None
                    282:         else:
                    283:             metaDatas=self.ZopeFind(self.aq_parent,obj_metatypes=['OSAS_Metadata__neu'],search_sub=1)
                    284:             if metaDatas:
                    285:                 metaData=metaDatas[0][1]
                    286:             else:
                    287:                 zLOG.LOG('OSAS_browser (readObjectsFromPath)',zLOG.INFO,"There is no OSAS_Metadata Object")
                    288:                 metaData=None
                    289:                 
                    290:         #print "md",metaData
1.1       dwinter   291:         if realPath.find(self.rootFolderName) <0: #versuch auf Pfad unterhalb des Rootfolder zuzugreifen
                    292:             return {}
                    293:             
                    294:         
1.8       dwinter   295:        
                    296:         
                    297:         stats=server.getStat(realPath)
                    298: 
                    299:         if not stats:
1.2       dwinter   300:             return None
                    301:         
1.3       dwinter   302:         #teste ob schon im cache
                    303:         if self._v_fileSystem.has_key(realPath) and (self._v_fileSystem[realPath][0]==stats[stat.ST_MTIME]):
1.1       dwinter   304:             
1.3       dwinter   305:             return self._v_fileSystem[realPath][1]
1.1       dwinter   306: 
1.8       dwinter   307:         dir=server.listdir(realPath)
1.1       dwinter   308:         ret={}
                    309:         for filename in dir:
                    310:             object=os.path.join(realPath,filename)
                    311:             fileType=OSAS_helpers.checkOSASFileType(object)
                    312:             
                    313:             if fileType:
1.7       dwinter   314:                 if metaData:
                    315:                  
                    316:                     ret[object]=(fileType,metaData.getDisplayFieldsAsStr(object))
                    317:                 else:
                    318:                     ret[object]=(fileType,'')
1.1       dwinter   319:             
                    320:         self._v_fileSystem[realPath]=(stats[stat.ST_MTIME],ret) # speicher im chache
                    321:         
                    322:         return ret
                    323: 
                    324:     def giveHandlers(self,path,type):
1.7       dwinter   325:         """teste ob fuer diesen Typ, handler definiert sind und gibt einen entsprechenden Link zurueck, der das Object mit diesem Handler ausfuehrt. Die Handler mussen im parent ordner des browser oder einem Subordner davon liegen. 
1.2       dwinter   326:         @param path: Pfad auf das Objekt
                    327:         @param type: Typ des Objektes
                    328:         @return: (string) html-Fragment, link der das Objekt mit diesem Handler anzeigt.
                    329:         """
1.1       dwinter   330:         ret=[]
                    331:         
1.4       dwinter   332:         for handler in self.ZopeFind(self.aq_parent,obj_metatypes=['OSAS_HandlerObject__neu'],search_sub=1):
                    333:             if type in handler[1].objectTypes:
1.6       dwinter   334:        try:
                    335:                  path=path.replace(getattr(handler[1],'ignorePath',''),'')
                    336:                 except:
                    337:           pass
1.4       dwinter   338:                 url=handler[1].prefix%path
                    339:                 text=handler[1].title
1.1       dwinter   340:                 string="""<a target="_blank" href="%s">%s</a>"""%(url,text)
                    341:                 ret.append(string)
                    342:         return ret
                    343:                       
                    344:         
                    345:     def generateTree(self,path=""):
1.2       dwinter   346:         """erzeuge liest die Objekte aus die im Pfad gespeichert sind
1.7       dwinter   347:         
                    348:         @keyword path: optional mit default='', Pfad relativ zu rootFolderName
1.2       dwinter   349:         @return: List von Tripeln, (link_html,array of handlers,metainformationen) hierbei ist
1.7       dwinter   350:          - (string) link_html ein html-Fragement, falls das Objekt vom Typ OSAS_dir ist, ist dies ein Link auf dieses Verzeichnis, sonst der Dateiname
                    351:          - (string) handler sind die Ergebnisse von giveHandlers fuer dieses Objekt
                    352:          - (string) metainformationen die Metainformationen zum Objekt als Ergebnis von readObjectsFromPath
1.2       dwinter   353:         """
1.1       dwinter   354:         objects=self.readObjectsFromPath(path)
1.2       dwinter   355:         if not objects:
                    356:             objects={}
1.9       dwinter   357: 
1.10    ! dwinter   358: 
        !           359:         im=self.getSubDirsFromIndexMeta(path)
1.2       dwinter   360:         for key in im.keys():
                    361:             #virtuelle pfade hinzufuegen
                    362:             
                    363:             if not objects.has_key(key):
                    364:                 objects[key]=im[key]
                    365:                 
                    366:         
1.1       dwinter   367:         def sortLow(x,y):
                    368:             return cmp(x.lower(),y.lower())
                    369:         
                    370:         ret=[]
                    371:         
                    372:         objectSorted=objects.keys()
                    373:         objectSorted.sort(sortLow)
                    374:         for object in objectSorted:
                    375:             handler=self.giveHandlers(object,objects[object][0])
                    376:             if objects[object][0]=="OSAS_dir":
                    377:                 
                    378:                 string="""<a href="?path=%s">%s</a>"""%(object,os.path.split(object)[1])
                    379:                 
                    380:                 ret.append((string,handler,objects[object][1]))
1.2       dwinter   381:             elif objects[object][0]=="OSAS_dir_archive":
                    382:                 string="""<a href="?path=%s">%s (A)</a>"""%(object,os.path.split(object)[1])
                    383:                 
                    384:                 ret.append((string,handler,objects[object][1]))
1.1       dwinter   385:             else:
                    386:                 ret.append((os.path.split(object)[1],handler,objects[object][1]))
                    387: 
                    388:                      
                    389:         return ret
                    390: 
                    391: 
                    392:     def path_to_link(self,pathTmp=""):
1.7       dwinter   393:         """generates navigation bar for viewfiles
                    394:         @keyword pathTmp: optional, generisch="", pfad der erstellt werden soll
                    395:         @return: html Fragment, pathTmp zerlegt, dass jeder Teil von Pfad unterhalb von rootFolderName direkt angesprunden werden kann.
                    396:         """
1.1       dwinter   397: 
                    398:         path=os.path.normpath(os.path.join(self.rootFolderName,pathTmp))
                    399:         
                    400:         URL=self.absolute_url()
                    401:         string=""
                    402:         
                    403:         tmppath=os.path.dirname(path)
                    404:         i=0
                    405:         pathes=[[path, os.path.basename(path)]]
                    406: 
                    407:         while not (len(tmppath)==1):
                    408: 
                    409:               i=i+1
                    410:               if i>20: break
                    411: 
                    412:               pathes.append([tmppath, os.path.basename(tmppath)])
                    413:               tmppath=os.path.dirname(tmppath)
                    414: 
                    415:         while i>=0:
                    416:             if pathes[i][0].find(self.rootFolderName) <0: #versuch auf Pfad unterhalb des Rootfolder zuzugreifen
                    417:                 string=string+"<a>"+pathes[i][1]+"</a>/"
                    418:             else:
                    419:                 string=string+"<a href="+URL+"?path="+pathes[i][0]+">"+pathes[i][1]+"</a>/"
                    420:             
                    421:             i=i-1
                    422:         return string
                    423: 
                    424: 
                    425: InitializeClass(OSAS_storeOnline)
                    426:    
                    427: def manage_addOSAS_storeOnlineForm(self):
1.7       dwinter   428:     """interface for adding the OSAS_storeOnline"""
1.1       dwinter   429:     pt=PageTemplateFile(os.path.join(package_home(globals()),'zpt','addStoreOnline.zpt')).__of__(self)
                    430:     return pt()
                    431: 
                    432: def manage_addOSAS_storeOnline(self,id,RESPONSE=None):
1.7       dwinter   433:     """add the OSAS_storeOnline
                    434:     @param id: id
                    435:     """
1.1       dwinter   436:     newObj=OSAS_storeOnline(id)
                    437:     self._setObject(id,newObj)
                    438:     if RESPONSE is not None:
                    439:         RESPONSE.redirect('manage_main')
                    440: 

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>