2009-06-29 07:06:01 -05:00
|
|
|
'''This file contains basic classes that will be added into any user
|
|
|
|
application for creating the basic structure of the application "Tool" which
|
|
|
|
is the set of web pages used for configuring the application. The "Tool" is
|
|
|
|
available to administrators under the standard Plone link "site setup". Plone
|
|
|
|
itself is shipped with several tools used for conguring the various parts of
|
|
|
|
Plone (content types, catalogs, workflows, etc.)'''
|
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
2010-11-10 08:15:00 -06:00
|
|
|
import types
|
2010-09-02 09:16:08 -05:00
|
|
|
from appy.gen import *
|
2009-06-29 07:06:01 -05:00
|
|
|
|
|
|
|
# ------------------------------------------------------------------------------
|
|
|
|
class ModelClass:
|
|
|
|
'''This class is the abstract class of all predefined application classes
|
2010-10-14 07:43:56 -05:00
|
|
|
used in the Appy model: Tool, User, etc. All methods and attributes of
|
|
|
|
those classes are part of the Appy machinery and are prefixed with _appy_
|
|
|
|
in order to avoid name conflicts with user-defined parts of the
|
|
|
|
application model.'''
|
2009-06-29 07:06:01 -05:00
|
|
|
_appy_attributes = [] # We need to keep track of attributes order.
|
2010-08-05 11:23:17 -05:00
|
|
|
# When creating a new instance of a ModelClass, the following attributes
|
|
|
|
# must not be given in the constructor (they are computed attributes).
|
2010-10-19 03:47:42 -05:00
|
|
|
_appy_notinit = ('id', 'type', 'pythonType', 'slaves', 'isSelect',
|
|
|
|
'hasLabel', 'hasDescr', 'hasHelp', 'master_css',
|
2011-01-14 02:06:25 -06:00
|
|
|
'required', 'filterable', 'validable', 'backd', 'isBack',
|
|
|
|
'sync', 'pageName')
|
2009-06-29 07:06:01 -05:00
|
|
|
|
2010-01-06 11:36:16 -06:00
|
|
|
@classmethod
|
2009-06-29 07:06:01 -05:00
|
|
|
def _appy_getTypeBody(klass, appyType):
|
|
|
|
'''This method returns the code declaration for p_appyType.'''
|
|
|
|
typeArgs = ''
|
|
|
|
for attrName, attrValue in appyType.__dict__.iteritems():
|
2011-01-14 02:06:25 -06:00
|
|
|
if attrName in ModelClass._appy_notinit: continue
|
|
|
|
if attrName == 'layouts':
|
|
|
|
if klass.__name__ == 'Tool': continue
|
|
|
|
# For Tool attributes we do not copy layout info. Indeed, most
|
|
|
|
# fields added to the Tool are config-related attributes whose
|
|
|
|
# layouts must be standard.
|
|
|
|
attrValue = appyType.getInputLayouts()
|
|
|
|
elif isinstance(attrValue, basestring):
|
2009-06-29 07:06:01 -05:00
|
|
|
attrValue = '"%s"' % attrValue
|
2010-08-05 11:23:17 -05:00
|
|
|
elif isinstance(attrValue, Ref):
|
2011-01-14 02:06:25 -06:00
|
|
|
if not attrValue.isBack: continue
|
|
|
|
attrValue = klass._appy_getTypeBody(attrValue)
|
2009-06-29 07:06:01 -05:00
|
|
|
elif type(attrValue) == type(ModelClass):
|
|
|
|
moduleName = attrValue.__module__
|
|
|
|
if moduleName.startswith('appy.gen'):
|
|
|
|
attrValue = attrValue.__name__
|
|
|
|
else:
|
|
|
|
attrValue = '%s.%s' % (moduleName, attrValue.__name__)
|
2010-01-06 11:36:16 -06:00
|
|
|
elif isinstance(attrValue, Selection):
|
|
|
|
attrValue = 'Selection("%s")' % attrValue.methodName
|
2010-08-05 11:23:17 -05:00
|
|
|
elif isinstance(attrValue, Group):
|
|
|
|
attrValue = 'Group("%s")' % attrValue.name
|
2010-10-19 03:47:42 -05:00
|
|
|
elif isinstance(attrValue, Page):
|
2011-01-14 02:06:25 -06:00
|
|
|
attrValue = 'pages["%s"]' % attrValue.name
|
|
|
|
elif callable(attrValue):
|
2010-08-05 11:23:17 -05:00
|
|
|
attrValue = '%sWrapper.%s'% (klass.__name__, attrValue.__name__)
|
2009-06-29 07:06:01 -05:00
|
|
|
typeArgs += '%s=%s,' % (attrName, attrValue)
|
|
|
|
return '%s(%s)' % (appyType.__class__.__name__, typeArgs)
|
|
|
|
|
2010-01-06 11:36:16 -06:00
|
|
|
@classmethod
|
2009-06-29 07:06:01 -05:00
|
|
|
def _appy_getBody(klass):
|
|
|
|
'''This method returns the code declaration of this class. We will dump
|
|
|
|
this in appyWrappers.py in the resulting product.'''
|
2011-01-14 02:06:25 -06:00
|
|
|
res = 'class %s(%sWrapper):\n' % (klass.__name__, klass.__name__)
|
|
|
|
if klass.__name__ == 'Tool':
|
|
|
|
res += ' folder=True\n'
|
|
|
|
# First, scan all attributes, determine all used pages and create a
|
|
|
|
# dict with it. It will prevent us from creating a new Page instance
|
|
|
|
# for every field.
|
|
|
|
pages = {}
|
|
|
|
for attrName in klass._appy_attributes:
|
|
|
|
exec 'appyType = klass.%s' % attrName
|
|
|
|
if appyType.page.name not in pages:
|
|
|
|
pages[appyType.page.name] = appyType.page
|
|
|
|
res += ' pages = {'
|
|
|
|
for page in pages.itervalues():
|
|
|
|
# Determine page show
|
|
|
|
pageShow = page.show
|
|
|
|
if isinstance(pageShow, basestring): pageShow='"%s"' % pageShow
|
|
|
|
res += '"%s":Page("%s", show=%s),'% (page.name, page.name, pageShow)
|
|
|
|
res += '}\n'
|
|
|
|
# Secondly, dump every attribute
|
2009-06-29 07:06:01 -05:00
|
|
|
for attrName in klass._appy_attributes:
|
|
|
|
exec 'appyType = klass.%s' % attrName
|
|
|
|
res += ' %s=%s\n' % (attrName, klass._appy_getTypeBody(appyType))
|
|
|
|
return res
|
|
|
|
|
2010-10-14 07:43:56 -05:00
|
|
|
# The User class ---------------------------------------------------------------
|
2010-09-02 09:16:08 -05:00
|
|
|
class User(ModelClass):
|
2010-10-14 07:43:56 -05:00
|
|
|
# In a ModelClass we need to declare attributes in the following list.
|
|
|
|
_appy_attributes = ['title', 'name', 'firstName', 'login', 'password1',
|
|
|
|
'password2', 'roles']
|
2010-09-02 09:16:08 -05:00
|
|
|
# All methods defined below are fake. Real versions are in the wrapper.
|
2010-12-06 04:11:40 -06:00
|
|
|
title = String(show=False, indexed=True)
|
2010-09-02 09:16:08 -05:00
|
|
|
gm = {'group': 'main', 'multiplicity': (1,1)}
|
|
|
|
name = String(**gm)
|
|
|
|
firstName = String(**gm)
|
|
|
|
def showLogin(self): pass
|
|
|
|
def validateLogin(self): pass
|
2010-09-13 14:04:10 -05:00
|
|
|
login = String(show=showLogin, validator=validateLogin, indexed=True, **gm)
|
2010-09-02 09:16:08 -05:00
|
|
|
def showPassword(self): pass
|
|
|
|
def validatePassword(self): pass
|
|
|
|
password1 = String(format=String.PASSWORD, show=showPassword,
|
|
|
|
validator=validatePassword, **gm)
|
|
|
|
password2 = String(format=String.PASSWORD, show=showPassword, **gm)
|
|
|
|
gm['multiplicity'] = (0, None)
|
2010-12-17 07:46:55 -06:00
|
|
|
roles = String(validator=Selection('getGrantableRoles'), indexed=True, **gm)
|
2010-09-02 09:16:08 -05:00
|
|
|
|
2011-01-14 02:06:25 -06:00
|
|
|
# The Translation class --------------------------------------------------------
|
|
|
|
class Translation(ModelClass):
|
|
|
|
_appy_attributes = ['po', 'title']
|
|
|
|
# All methods defined below are fake. Real versions are in the wrapper.
|
|
|
|
def getPoFile(self): pass
|
|
|
|
po = Action(action=getPoFile, page=Page('actions', show='view'),
|
2011-02-12 10:09:11 -06:00
|
|
|
result='filetmp')
|
2011-01-14 02:06:25 -06:00
|
|
|
title = String(show=False, indexed=True)
|
|
|
|
def computeLabel(self): pass
|
|
|
|
def showField(self, name): pass
|
2009-06-29 07:06:01 -05:00
|
|
|
|
2011-01-14 02:06:25 -06:00
|
|
|
# The Tool class ---------------------------------------------------------------
|
2010-10-14 07:43:56 -05:00
|
|
|
# Here are the prefixes of the fields generated on the Tool.
|
|
|
|
toolFieldPrefixes = ('defaultValue', 'podTemplate', 'formats', 'resultColumns',
|
|
|
|
'enableAdvancedSearch', 'numberOfSearchColumns',
|
|
|
|
'searchFields', 'optionalFields', 'showWorkflow',
|
|
|
|
'showWorkflowCommentField', 'showAllStatesInPhase')
|
2011-01-14 02:06:25 -06:00
|
|
|
defaultToolFields = ('users', 'translations', 'enableNotifications',
|
|
|
|
'unoEnabledPython', 'openOfficePort',
|
|
|
|
'numberOfResultsPerPage', 'listBoxesMaximumWidth')
|
2009-06-29 07:06:01 -05:00
|
|
|
|
2010-10-14 07:43:56 -05:00
|
|
|
class Tool(ModelClass):
|
|
|
|
# In a ModelClass we need to declare attributes in the following list.
|
|
|
|
_appy_attributes = list(defaultToolFields)
|
|
|
|
|
|
|
|
# Tool attributes
|
2011-01-14 02:06:25 -06:00
|
|
|
def validPythonWithUno(self, value): pass # Real method in the wrapper
|
|
|
|
unoEnabledPython = String(group="connectionToOpenOffice",
|
|
|
|
validator=validPythonWithUno)
|
|
|
|
openOfficePort = Integer(default=2002, group="connectionToOpenOffice")
|
|
|
|
numberOfResultsPerPage = Integer(default=30)
|
|
|
|
listBoxesMaximumWidth = Integer(default=100)
|
2010-10-14 07:43:56 -05:00
|
|
|
# First arg of Ref field below is None because we don't know yet if it will
|
|
|
|
# link to the predefined User class or a custom class defined in the
|
|
|
|
# application.
|
|
|
|
users = Ref(None, multiplicity=(0,None), add=True, link=False,
|
2010-12-06 04:11:40 -06:00
|
|
|
back=Ref(attribute='toTool'), page='users', queryable=True,
|
|
|
|
queryFields=('login',), showHeaders=True,
|
|
|
|
shownInfo=('login', 'title', 'roles'))
|
2011-01-14 02:06:25 -06:00
|
|
|
translations = Ref(Translation, multiplicity=(0,None), add=False,link=False,
|
|
|
|
back=Ref(attribute='trToTool', show=False), show='view',
|
|
|
|
page=Page('translations', show='view'))
|
2010-10-14 07:43:56 -05:00
|
|
|
enableNotifications = Boolean(default=True, page='notifications')
|
2010-01-06 11:36:16 -06:00
|
|
|
|
|
|
|
@classmethod
|
2009-06-29 07:06:01 -05:00
|
|
|
def _appy_clean(klass):
|
|
|
|
toClean = []
|
|
|
|
for k, v in klass.__dict__.iteritems():
|
|
|
|
if not k.startswith('__') and (not k.startswith('_appy_')):
|
2010-10-14 07:43:56 -05:00
|
|
|
if k not in defaultToolFields:
|
2009-06-29 07:06:01 -05:00
|
|
|
toClean.append(k)
|
|
|
|
for k in toClean:
|
|
|
|
exec 'del klass.%s' % k
|
2010-10-14 07:43:56 -05:00
|
|
|
klass._appy_attributes = list(defaultToolFields)
|
2009-06-29 07:06:01 -05:00
|
|
|
# ------------------------------------------------------------------------------
|