Annotation of OSA_system2/OSAS_browser.py, revision 1.22

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
1.22    ! dwinter    11: 
        !            12: import logging
        !            13: 
        !            14: #ersetzt logging
        !            15: def logger(txt,method,txt2):
        !            16:     """logging"""
        !            17:     logging.info(txt+ txt2)
        !            18: 
        !            19: 
1.15      dwinter    20: import base64
1.14      dwinter    21: import bz2
1.1       dwinter    22: import os
                     23: import os.path
                     24: import stat
1.21      dwinter    25: import Ft.Xml.XPath
1.2       dwinter    26: import xml.dom.minidom
1.3       dwinter    27: from types import *
1.8       dwinter    28: import xmlrpclib
1.10      dwinter    29: from OSAS_helpers import *
1.8       dwinter    30: 
1.15      dwinter    31: def decodeRPC(string):
                     32:     return bz2.decompress(base64.decodestring(string))
                     33: 
1.1       dwinter    34: 
                     35: class OSAS_storeOnline(SimpleItem):
                     36:     """Webfrontend für das Storagesystem
                     37:     liefert Browserumgebung 
                     38:     """
                     39:     meta_type="OSAS_StoreOnline__neu"
                     40:     
                     41:     security=ClassSecurityInfo()
                     42: 
                     43:     _v_fileSystem={} #chache fuer filesystem
1.10      dwinter    44:     _v_metaFiles={} #chache fuer indexMeta
1.15      dwinter    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: 
1.16      dwinter    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:                
1.10      dwinter    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:
1.21      dwinter    90:                   
1.10      dwinter    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"""
1.8       dwinter   135: 
1.10      dwinter   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:         """
1.16      dwinter   146: 
1.10      dwinter   147:         xmlInfos=self.findEntryInIndexMeta(path)
1.16      dwinter   148: 
1.10      dwinter   149:         if xmlInfos:
                    150:             return OSAS_helpers.getMetaInfoFromXML(path,xmlInfos)
                    151:         else:
                    152:             return ""
1.8       dwinter   153:    
1.10      dwinter   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
1.16      dwinter   162: 
1.10      dwinter   163:         if not indexMeta:
                    164:             return None
                    165: 
                    166:         realPath=os.path.split(indexMeta)[0]
                    167:         path=os.path.normpath(path)
                    168: 
                    169:         try:
1.21      dwinter   170:             dom = NonvalidatingReader.parseString(server.getFile(indexMeta),"http://www.mpiwg-berlin.mpg.de/")
                    171:            
1.10      dwinter   172:         except:
1.22    ! dwinter   173:             logger("OSAS_browser (findEntryInIndexMeta)",logging.ERROR,"Cannot parse: %s"%indexMeta)
1.15      dwinter   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:]
1.10      dwinter   183:         #ist path ein directory? 
1.15      dwinter   184:         xpath="/resource/dir[name='%s' and path='%s']"%(name,searchPath)
1.21      dwinter   185:         dirs=Ft.Xml.XPath.Evaluate(xpath,contextNode=dom)
1.15      dwinter   186: 
1.10      dwinter   187: 
1.15      dwinter   188:         if len(dirs)>0:
                    189:             return dirs[0].toxml
1.10      dwinter   190: 
1.15      dwinter   191:         #ist path ein file?      
                    192:         xpath="/resource/file[name='%s' and path='%s']"%(name,searchPath)
1.10      dwinter   193: 
                    194: 
1.21      dwinter   195:         dirs=Ft.Xml.XPath.Evaluate(xpath,contextNode=dom)
1.15      dwinter   196:         if len(dirs)>0:
                    197:             return dirs[0].toxml()
1.10      dwinter   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={}
1.17      dwinter   210:         startPath=path
1.10      dwinter   211:         server=xmlrpclib.Server(self.serverUrl)
                    212:         indexMeta,stats=server.findIndexMetaWithStats(path)#findex index.meta zu path.
1.16      dwinter   213: 
1.10      dwinter   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 lösche dieses.
                    220:             path=path[1:]
                    221: 
                    222: 
                    223: 
1.17      dwinter   224:         #teste ob schon im cache zur Zeit kein chache wenn index.meta file nicht im selben ordner wie path.
                    225: 
1.20      casties   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]
1.10      dwinter   229: 
                    230:         try:
                    231:             dom=xml.dom.minidom.parseString(server.getFile(indexMeta))
                    232:         except:
1.22    ! dwinter   233:             logger("OSAS_browser (getSubDirsFromIndexMeta)",logging.ERROR,"Cannot parse: %s"%indexMeta)
1.10      dwinter   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: 
1.17      dwinter   261:         self._v_metaFiles[startPath]=(stats[stat.ST_MTIME],ret) # speicher im chache
1.10      dwinter   262:       
                    263:         return ret
                    264: 
                    265:         
1.1       dwinter   266:     
1.8       dwinter   267:     def __init__(self,id,serverUrl):
1.7       dwinter   268:         """initialize a new instance
                    269:         @param id: Zope id"""
1.1       dwinter   270:         self.id = id
1.8       dwinter   271:         self.serverUrl = serverUrl
1.1       dwinter   272:     
                    273: 
                    274:     security.declareProtected('View','index_html')
                    275:     def index_html(self):
1.2       dwinter   276:         """main view either standard template zpt/storeOnline_index_html.zpt or storeOnline_index.html in tree"""
1.1       dwinter   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: 
1.2       dwinter   284:     def findIndexMeta(self,path=""):
1.7       dwinter   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
1.2       dwinter   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
1.8       dwinter   291:         server=xmlrpclib.Server(self.serverUrl)
                    292:         return server.findIndexMeta(realPath)
1.7       dwinter   293:         
1.2       dwinter   294: 
1.8       dwinter   295:   
1.7       dwinter   296:     def readObjectsFromPath(self,path="",metaDataId=None):
1.1       dwinter   297:         """Liest files aus dem path und speichert im cache _v_filesystem.
                    298: 
1.7       dwinter   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.
1.2       dwinter   301:         @return: directory der Form [pfad zum Objekt] -> (fileType,metadatum als String)
1.1       dwinter   302:         """
1.8       dwinter   303:         server=xmlrpclib.Server(self.serverUrl)             
1.1       dwinter   304:         realPath=os.path.normpath(os.path.join(self.rootFolderName,path))
1.7       dwinter   305: 
                    306:         if metaDataId:
                    307:             metaData=getattr(self,metaDataId)
                    308:             if not (getattr(metaData,'meta_type','')=='OSAS_Metadata__neu'):
1.22    ! dwinter   309:                 logger('OSAS_browser (readObjectsFromPath)',logging.ERROR,"%s is not OSAS_Metadata")
1.7       dwinter   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:
1.22    ! dwinter   316:                 logger('OSAS_browser (readObjectsFromPath)',logging.INFO,"There is no OSAS_Metadata Object")
1.7       dwinter   317:                 metaData=None
                    318:                 
                    319:         #print "md",metaData
1.1       dwinter   320:         if realPath.find(self.rootFolderName) <0: #versuch auf Pfad unterhalb des Rootfolder zuzugreifen
                    321:             return {}
                    322:             
                    323:         
1.8       dwinter   324:        
                    325:         
                    326:         stats=server.getStat(realPath)
                    327: 
                    328:         if not stats:
1.2       dwinter   329:             return None
                    330:         
1.3       dwinter   331:         #teste ob schon im cache
1.20      casties   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]
1.15      dwinter   335:         
                    336:         indexMetas=server.getAllIndexMetasOfSubDirs(realPath)
1.13      dwinter   337:         dir=indexMetas.keys()
1.11      dwinter   338:         
1.1       dwinter   339:         ret={}
                    340:         for filename in dir:
1.16      dwinter   341: 
1.1       dwinter   342:             object=os.path.join(realPath,filename)
1.13      dwinter   343:             fileType=indexMetas[filename][0]
1.15      dwinter   344:             
1.1       dwinter   345:             if fileType:
1.15      dwinter   346:                 if (fileType=='OSAS_dir') and indexMetas.has_key(".") and indexMetas["."][1]:
1.16      dwinter   347: 
1.15      dwinter   348:                     if(OSAS_helpers.isImageFolder(object,decodeRPC(indexMetas["."][1]))):
1.13      dwinter   349:                         fileType='OSAS_imageFolder'
1.15      dwinter   350:                     elif(OSAS_helpers.isVideoFolder(object,decodeRPC(indexMetas["."][1]))):
                    351:                         fileType='OSAS_videoFolder'
                    352:                 if metaData and indexMetas[filename][1]:
1.16      dwinter   353: 
1.15      dwinter   354:                     ret[object]=(fileType,metaData.getDisplayFieldsAsStr(decodeRPC(indexMetas[filename][1])))
1.7       dwinter   355:                 else:
1.15      dwinter   356:                     metaDataStr=self.findEntryInIndexMeta(object)
1.16      dwinter   357: 
1.15      dwinter   358:                     if metaDataStr:
                    359:                         display=metaData.getDisplayFieldsAsStr(metaDataStr)
                    360: 
1.21      dwinter   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:
1.15      dwinter   363:                             fileType='OSAS_videoFile'
                    364:                             
                    365:                     else:
                    366:                         display=""
1.16      dwinter   367: 
1.15      dwinter   368:                     
                    369:                     ret[object]=(fileType,display)
1.1       dwinter   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):
1.7       dwinter   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. 
1.2       dwinter   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:         """
1.1       dwinter   381:         ret=[]
                    382:         
1.4       dwinter   383:         for handler in self.ZopeFind(self.aq_parent,obj_metatypes=['OSAS_HandlerObject__neu'],search_sub=1):
1.15      dwinter   384:             
1.4       dwinter   385:             if type in handler[1].objectTypes:
1.19      dwinter   386:                 try:
1.6       dwinter   387:                  path=path.replace(getattr(handler[1],'ignorePath',''),'')
                    388:                 except:
1.19      dwinter   389:                    pass
1.4       dwinter   390:                 url=handler[1].prefix%path
                    391:                 text=handler[1].title
1.1       dwinter   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=""):
1.2       dwinter   398:         """erzeuge liest die Objekte aus die im Pfad gespeichert sind
1.7       dwinter   399:         
                    400:         @keyword path: optional mit default='', Pfad relativ zu rootFolderName
1.2       dwinter   401:         @return: List von Tripeln, (link_html,array of handlers,metainformationen) hierbei ist
1.7       dwinter   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
1.2       dwinter   405:         """
1.17      dwinter   406: 
                    407:         objects=self.getSubDirsFromIndexMeta(path)
                    408: 
1.18      dwinter   409:         
1.17      dwinter   410:         
                    411:         im=self.readObjectsFromPath(path)
                    412:         if not im:
                    413:             im={}
1.9       dwinter   414: 
1.10      dwinter   415: 
1.17      dwinter   416: 
                    417:         
1.2       dwinter   418:         for key in im.keys():
1.17      dwinter   419:             #relle  pfade hinzufuegen, virtueller wird ueberschrieben
                    420:             
                    421:             objects[key]=im[key]
1.2       dwinter   422:             
1.17      dwinter   423:         
1.2       dwinter   424:         
1.1       dwinter   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:
1.15      dwinter   433:             
1.1       dwinter   434:             handler=self.giveHandlers(object,objects[object][0])
1.16      dwinter   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]))
1.1       dwinter   448: 
                    449:                      
                    450:         return ret
                    451: 
                    452: 
                    453:     def path_to_link(self,pathTmp=""):
1.7       dwinter   454:         """generates navigation bar for viewfiles
                    455:         @keyword pathTmp: optional, generisch="", pfad der erstellt werden soll
1.18      dwinter   456:         @return: html Fragment, pathTmp zerlegt, dass jeder Teil von Pfad unterhalb von rootFolderName direkt angesprungen werden kann.
1.7       dwinter   457:         """
1.1       dwinter   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)
1.19      dwinter   487:         
1.1       dwinter   488: def manage_addOSAS_storeOnlineForm(self):
1.7       dwinter   489:     """interface for adding the OSAS_storeOnline"""
1.1       dwinter   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):
1.7       dwinter   494:     """add the OSAS_storeOnline
                    495:     @param id: id
                    496:     """
1.1       dwinter   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>