File:  [Repository] / OSA_system2 / OSAS_browser.py
Revision 1.23: download - view: text, annotated - select for diffs - revision graph
Mon Feb 15 19:09:10 2010 UTC (14 years, 4 months ago) by casties
Branches: MAIN
CVS tags: HEAD
fixing small errors for zope 2.12

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

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