"""
This module handles requests from the application for PSP pages.
--------------------------------------------------------------------------
(c) Copyright by Jay Love, 2000 (mailto:jsliv@jslove.net)
Permission to use, copy, modify, and distribute this software and its
documentation for any purpose and without fee or royalty is hereby granted,
provided that the above copyright notice appear in all copies and that
both that copyright notice and this permission notice appear in
supporting documentation or portions thereof, including modifications,
that you make.
THE AUTHORS DISCLAIM ALL WARRANTIES WITH REGARD TO
THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
FITNESS, IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL,
INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING
FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION
WITH THE USE OR PERFORMANCE OF THIS SOFTWARE !
"""
from WebKit.ServletFactory import ServletFactory
import string
import os,sys,string
from PSP import Context, PSPCompiler
import time, threading
class PSPServletFactory(ServletFactory):
"""
Servlet Factory for PSP files.
Very sloppy. Need to come back and do a serious cleanup.
"""
def __init__(self,application):
ServletFactory.__init__(self,application)
self.cacheDir = application.serverSidePath('Cache/PSP')
self._classcache={}
sys.path.append(self.cacheDir)
self._cacheClassFiles = self._cacheClasses
l = ['_'] * 256
for c in string.digits + string.letters:
l[ord(c)] = c
self._classNameTrans = string.join(l, '')
if application.setting('ClearPSPCacheOnStart', 1):
self.clearFileCache()
self._lock = threading.RLock()
def uniqueness(self):
return 'file'
def extensions(self):
return ['.psp']
def flushCache(self):
"""
Clean out the cache of classes we keep in memory. Also, clear the class files stored on disk.
"""
self._classcache = {}
self.clearFileCache()
def clearFileCache(self):
"""
Clear class files stored on disk.
"""
import glob
files = glob.glob(os.path.join(self.cacheDir,'*.*'))
map(os.remove, files)
def computeClassName(self,pagename):
"""
Generates a <hopefully> unique class/file name for each PSP file.
Argument: pagename: the path to the PSP source file
Returns: A unique name for the class generated fom this PSP source file.
"""
# Compute class name by taking the path and substituting underscores for
# all non-alphanumeric characters.
return string.translate(os.path.splitdrive(pagename)[1], self._classNameTrans)
# def import_createInstanceFromFile(self,transaction,path,classname,mtime,reimp=0):
# """
# Create an actual instance of a PSP class. This version uses import to generate the instance.
#
# """
# globals={}
# module_obj=__import__(classname)
# if reimp:
# reload(module_obj)
# instance = module_obj.__dict__[classname]()
# code=module_obj.__dict__[classname]
# self._classcache[classname] = {'code':code,
# 'filename':path,
# 'mtime':time.time(),}
# return instance
# def createInstanceFromFile(self,transaction,filename,classname,mtime,reimp=0):
# """
# Create an actual class instance. This version uses "exec" to generate the class instance.
# """
# globals={}
# execfile(filename,globals)
# assert globals.has_key(classname)
# instance = globals[classname]()
# code=globals[classname]
# self._classcache[classname] = {'code':code,
# 'filename':filename,
# 'mtime':time.time(),}
# return instance
def createInstanceFromFile(self,transaction,filename,classname,mtime,reimp=0):
"""
Create an actual class instance. The module containing the class is imported as though it
were a module within the context's package (and appropriate subpackages).
"""
module = self.importAsPackage(transaction,filename)
assert module.__dict__.has_key(classname), 'Cannot find expected class named %s in %s.' % (repr(classname), repr(filename))
code = getattr(module, classname)
instance = code()
self._classcache[classname] = {'code':code,
'filename':filename,
'mtime':time.time(),}
return instance
def checkClassCache(self, classname, mtime):
"""
Check our cache to see if we already have this class in memory.
"""
if self._classcache.has_key(classname) and self._classcache[classname]['mtime'] > mtime:
return self._classcache[classname]['code']()
return None
def servletForTransaction(self, trans):
"""
The entry point to getting a PSP servlet instance.
"""
fullname = trans.request().serverSidePath()
path,pagename = os.path.split(fullname)
mtime = os.path.getmtime(fullname)
instance = None
classname = self.computeClassName(fullname)
# Use a lock to prevent multiple simultaneous compilations/imports of the same PSP
self._lock.acquire()
try:
#see if we can just create a new instance
if self._cacheClasses:
instance = self.checkClassCache(classname,mtime)
if instance != None:
return instance
cachedfilename = os.path.join(self.cacheDir,str(classname + '.py'))
if self._cacheClassFiles and os.path.exists(cachedfilename) and os.stat(cachedfilename)[6] > 0:
if os.path.getmtime(cachedfilename) > mtime:
instance = self.createInstanceFromFile(trans,cachedfilename,classname,mtime,0)
return instance
pythonfilename = cachedfilename
context = Context.PSPCLContext(fullname,trans)
context.setClassName(classname)
context.setPythonFileName(pythonfilename)
clc = PSPCompiler.Compiler(context)
#print 'creating python class: ' , classname
clc.compile()
instance = self.createInstanceFromFile(trans,cachedfilename,classname,mtime,1)
return instance
finally:
self._lock.release()