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
15: import xml.dom.minidom
16: from types import *
17:
18: class OSAS_storeOnline(SimpleItem):
19: """Webfrontend für das Storagesystem
20: liefert Browserumgebung
21: """
22: meta_type="OSAS_StoreOnline__neu"
23:
24: security=ClassSecurityInfo()
25:
26: _v_fileSystem={} #chache fuer filesystem
27: _v_metaFiles={} #chache fuer indexMeta
28:
29: def __init__(self,id):
30: """initialize a new instance
31: @param id: Zope id"""
32: self.id = id
33:
34:
35:
36: security.declareProtected('View','index_html')
37: def index_html(self):
38: """main view either standard template zpt/storeOnline_index_html.zpt or storeOnline_index.html in tree"""
39: if hasattr(self,'storeOnline_index.html'):
40: return getattr(self,'storeOnline_index.html')()
41: else:
42: pt=PageTemplateFile(os.path.join(package_home(globals()),'zpt','storeOnline_index_html.zpt')).__of__(self)
43: return pt()
44:
45:
46: def findIndexMeta(self,path=""):
47: """finde Rueckwaerts im Baum von Pfad ausgehend, dass erste index.meta file
48: @keyword path: default ist "", Pfad auf das Object relativ zum rootFolderName
49: @return: None falls kein index.meta existiert sonst Pfad auf das index.meta
50: """
51: realPath=os.path.normpath(os.path.join(self.rootFolderName,path))
52: #suche index.meta
53: while (not os.path.exists(os.path.join(realPath,'index.meta'))) and (not ((realPath=="") or (realPath=="/"))):
54: realPath=os.path.split(realPath)[0]
55:
56: if realPath=='' or realPath=='/':
57: if os.path.exists(os.path.join(realPath,'index.meta')):
58: return os.path.join(realPath,'index.meta')
59: else:
60: return None
61: else:
62: return os.path.join(realPath,'index.meta')
63:
64: def findEntryInIndexMeta(self,path):
65: """Finde im naechstgelegenden index.meta relativ zu path den entprechenden Eintrag fuer diesen Pfad.
66: @param path: Pfad auf das Object relativ zum rootFolderName
67: @return: den Teil von Index.meta der Informationen zu path enthaelt, None wenn error.
68: """
69:
70: indexMeta=self.findIndexMeta(path) # suche index.meta
71:
72: if not indexMeta:
73: return None
74:
75: realPath=os.path.split(indexMeta)[0]
76: path=os.path.normpath(os.path.join(self.rootFolderName,path))
77:
78: try:
79: dom=xml.dom.minidom.parse(indexMeta)
80: except:
81: zLOG.LOG("OSAS_browser (findEntryInIndexMeta)",zLOG.ERROR,"Cannot parse: %s"%indexMeta)
82: #ist path ein directory?
83: dirs=dom.getElementsByTagName('dir')
84: for dir in dirs:
85: pathes=dir.getElementsByTagName('path')
86: if pathes:
87: pathX=OSAS_helpers.getText(pathes[0].childNodes)
88: else:
89: pathX=""
90: names=dir.getElementsByTagName('name')
91: if names:
92: name=OSAS_helpers.getText(names[0].childNodes)
93: else:
94: name=""
95:
96: checkpath=os.path.normpath(os.path.join(realPath,pathX,name))
97: if checkpath==path:
98:
99: return dir.toxml()
100:
101: #ist path ein file?
102: files=dom.getElementsByTagName('file')
103: for dir in dirs:
104: pathes=dir.getElementsByTagName('path')
105: if pathes:
106: pathX=OSAS_helpers.getText(pathes[0].childNodes)
107: else:
108: pathX=""
109: names=dir.getElementsByTagName('name')
110: if names:
111: name=OSAS_helpers.getText(names[0].childNodes)
112: else:
113: name=""
114:
115: checkpath=os.path.normpath(os.path.join(realPath,pathX,name))
116: if checkpath==path:
117:
118: return dir.toxml()
119:
120:
121: return None
122:
123: def getSubDirsFromIndexMeta(self,path):
124:
125: """Gebe alle path untergeordenten Objekte aus
126: @param path: optional, default ist "", Pfad auf das Object relativ zum rootFolderName
127: @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.
128: @todo: Rueckgabe einer Beschreibung gemaess Metadaten
129: """
130: ret={}
131: indexMeta=self.findIndexMeta(path)#findex index.meta zu path.
132:
133: if not indexMeta:
134: return ret
135:
136: realPath=os.path.split(indexMeta)[0]
137: path=path.replace(realPath,"")
138: if path and (path[0]==os.sep): #falls am Anfang os.sep steht lösche dieses.
139: path=path[1:]
140:
141:
142:
143: #teste ob schon im cache
144: stats=os.stat(indexMeta)
145: if self._v_metaFiles.has_key(path) and (self._v_metaFiles[path][0]==stats[stat.ST_MTIME]):
146: return self._v_metaFiles[path][1]
147:
148: try:
149: dom=xml.dom.minidom.parse(indexMeta)
150: except:
151: zLOG.LOG("OSAS_browser (getSubDirsFromIndexMeta)",zLOG.ERROR,"Cannot parse: %s"%indexMeta)
152:
153:
154: dirs=[]
155: dirs=dom.getElementsByTagName('dir')+dom.getElementsByTagName('file')
156:
157: for dir in dirs:
158: pathes=dir.getElementsByTagName('path')
159: if pathes:
160: pathX=OSAS_helpers.getText(pathes[0].childNodes)
161: else:
162: pathX=""
163: names=dir.getElementsByTagName('name')
164: if names:
165: name=OSAS_helpers.getText(names[0].childNodes)
166: else:
167: name=""
168:
169: #print "PP",pathX,path
170: if pathX==path:
171: if dir.tagName=="dir":
172: fileType="OSAS_dir_archive"
173: else:
174: fileType="OSAS_file_archive"
175:
176: object=os.path.join(realPath,pathX,name)
177: ret[object]=(fileType,'')
178:
179: self._v_metaFiles[path]=(stats[stat.ST_MTIME],ret) # speicher im chache
180: return ret
181:
182:
183: def getMetaInfoFromIndexMeta(self,path):
184: """metadaten zu path als html aus dem index.meta file zu path (meta tag im file bzw. dir container)
185: @param path: Pfad auf das Object relativ zum rootFolderName
186: @return: metadata als html
187: """
188: xmlInfos=self.findEntryInIndexMeta(path)
189: if xmlInfos:
190: return OSAS_helpers.getMetaInfoFromXML(path,xmlInfos)
191: else:
192: return ""
193:
194:
195: def readObjectsFromPath(self,path="",metaDataId=None):
196: """Liest files aus dem path und speichert im cache _v_filesystem.
197:
198: @keyword path : path relativ zum root folder des Storagesystems
199: @keyword metaDataId: Optional, id des OSAS_Metadata Object, dass benutzt werden soll, generisch wird das erste Object, dass in parent gefunden wird angezeigt.
200: @return: directory der Form [pfad zum Objekt] -> (fileType,metadatum als String)
201: """
202:
203: realPath=os.path.normpath(os.path.join(self.rootFolderName,path))
204:
205: if metaDataId:
206: metaData=getattr(self,metaDataId)
207: if not (getattr(metaData,'meta_type','')=='OSAS_Metadata__neu'):
208: zLOG.LOG('OSAS_browser (readObjectsFromPath)',zLOG.ERROR,"%s is not OSAS_Metadata")
209: metaData=None
210: else:
211: metaDatas=self.ZopeFind(self.aq_parent,obj_metatypes=['OSAS_Metadata__neu'],search_sub=1)
212: if metaDatas:
213: metaData=metaDatas[0][1]
214: else:
215: zLOG.LOG('OSAS_browser (readObjectsFromPath)',zLOG.INFO,"There is no OSAS_Metadata Object")
216: metaData=None
217:
218: #print "md",metaData
219: if realPath.find(self.rootFolderName) <0: #versuch auf Pfad unterhalb des Rootfolder zuzugreifen
220: return {}
221:
222:
223: if not os.path.exists(realPath):
224: #return None,"(ERROR) path %s does not exist."%path
225: return None
226:
227: stats=os.stat(realPath)
228:
229: #teste ob schon im cache
230: if self._v_fileSystem.has_key(realPath) and (self._v_fileSystem[realPath][0]==stats[stat.ST_MTIME]):
231:
232: return self._v_fileSystem[realPath][1]
233:
234: dir=os.listdir(realPath)
235: ret={}
236: for filename in dir:
237: object=os.path.join(realPath,filename)
238: fileType=OSAS_helpers.checkOSASFileType(object)
239:
240: if fileType:
241: if metaData:
242:
243: ret[object]=(fileType,metaData.getDisplayFieldsAsStr(object))
244: else:
245: ret[object]=(fileType,'')
246:
247: self._v_fileSystem[realPath]=(stats[stat.ST_MTIME],ret) # speicher im chache
248:
249: return ret
250:
251: def giveHandlers(self,path,type):
252: """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.
253: @param path: Pfad auf das Objekt
254: @param type: Typ des Objektes
255: @return: (string) html-Fragment, link der das Objekt mit diesem Handler anzeigt.
256: """
257: ret=[]
258:
259: for handler in self.ZopeFind(self.aq_parent,obj_metatypes=['OSAS_HandlerObject__neu'],search_sub=1):
260: if type in handler[1].objectTypes:
261: try:
262: path=path.replace(getattr(handler[1],'ignorePath',''),'')
263: except:
264: pass
265: url=handler[1].prefix%path
266: text=handler[1].title
267: string="""<a target="_blank" href="%s">%s</a>"""%(url,text)
268: ret.append(string)
269: return ret
270:
271:
272: def generateTree(self,path=""):
273: """erzeuge liest die Objekte aus die im Pfad gespeichert sind
274:
275: @keyword path: optional mit default='', Pfad relativ zu rootFolderName
276: @return: List von Tripeln, (link_html,array of handlers,metainformationen) hierbei ist
277: - (string) link_html ein html-Fragement, falls das Objekt vom Typ OSAS_dir ist, ist dies ein Link auf dieses Verzeichnis, sonst der Dateiname
278: - (string) handler sind die Ergebnisse von giveHandlers fuer dieses Objekt
279: - (string) metainformationen die Metainformationen zum Objekt als Ergebnis von readObjectsFromPath
280: """
281: objects=self.readObjectsFromPath(path)
282: if not objects:
283: objects={}
284: im=self.getSubDirsFromIndexMeta(path)
285: for key in im.keys():
286: #virtuelle pfade hinzufuegen
287:
288: if not objects.has_key(key):
289: objects[key]=im[key]
290:
291:
292: def sortLow(x,y):
293: return cmp(x.lower(),y.lower())
294:
295: ret=[]
296:
297: objectSorted=objects.keys()
298: objectSorted.sort(sortLow)
299: for object in objectSorted:
300: handler=self.giveHandlers(object,objects[object][0])
301: if objects[object][0]=="OSAS_dir":
302:
303: string="""<a href="?path=%s">%s</a>"""%(object,os.path.split(object)[1])
304:
305: ret.append((string,handler,objects[object][1]))
306: elif objects[object][0]=="OSAS_dir_archive":
307: string="""<a href="?path=%s">%s (A)</a>"""%(object,os.path.split(object)[1])
308:
309: ret.append((string,handler,objects[object][1]))
310: else:
311: ret.append((os.path.split(object)[1],handler,objects[object][1]))
312:
313:
314: return ret
315:
316:
317: def path_to_link(self,pathTmp=""):
318: """generates navigation bar for viewfiles
319: @keyword pathTmp: optional, generisch="", pfad der erstellt werden soll
320: @return: html Fragment, pathTmp zerlegt, dass jeder Teil von Pfad unterhalb von rootFolderName direkt angesprunden werden kann.
321: """
322:
323: path=os.path.normpath(os.path.join(self.rootFolderName,pathTmp))
324:
325: URL=self.absolute_url()
326: string=""
327:
328: tmppath=os.path.dirname(path)
329: i=0
330: pathes=[[path, os.path.basename(path)]]
331:
332: while not (len(tmppath)==1):
333:
334: i=i+1
335: if i>20: break
336:
337: pathes.append([tmppath, os.path.basename(tmppath)])
338: tmppath=os.path.dirname(tmppath)
339:
340: while i>=0:
341: if pathes[i][0].find(self.rootFolderName) <0: #versuch auf Pfad unterhalb des Rootfolder zuzugreifen
342: string=string+"<a>"+pathes[i][1]+"</a>/"
343: else:
344: string=string+"<a href="+URL+"?path="+pathes[i][0]+">"+pathes[i][1]+"</a>/"
345:
346: i=i-1
347: return string
348:
349: def getMetaFile(self,path):
350: """Lies Metafile ein
351: @param path: Pfad des index.met
352: @return: index.meta file
353: """
354: tmp=OSAS_helpers.getMetaFile(self,path)
355: #zLOG.LOG("EE",zLOG.INFO,type(tmp))
356:
357: return tmp
358:
359: InitializeClass(OSAS_storeOnline)
360:
361: def manage_addOSAS_storeOnlineForm(self):
362: """interface for adding the OSAS_storeOnline"""
363: pt=PageTemplateFile(os.path.join(package_home(globals()),'zpt','addStoreOnline.zpt')).__of__(self)
364: return pt()
365:
366: def manage_addOSAS_storeOnline(self,id,RESPONSE=None):
367: """add the OSAS_storeOnline
368: @param id: id
369: """
370: newObj=OSAS_storeOnline(id)
371: self._setObject(id,newObj)
372: if RESPONSE is not None:
373: RESPONSE.redirect('manage_main')
374:
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>