Commit d8635361 authored by Vincent Pelletier's avatar Vincent Pelletier Committed by Your Name

WorkflowTool: Accelerate action generation.

Include worklist parameter generation in the scope of existing cache.
Otherwise, it will be generated in pure loss if it is followed by a cache
hit. Most of WorkflowTool change is just indentation change.

Also, do some minor optimisations/simplifications in patchess.DCWorkflow.
Some comments:
- Guard_checkWithoutRoles return value is evaluated as a boolean, so no
  need to cast to int before returning based on boolean evaluation...
- DCWorkflowDefinition.worklists is always true, even when empty.
- Listing portal types per workflow requires checking all workflows, so
  build the whole mapping and cache it instead of caching for each workflow
  type individually (many more cache hits, fewer redundant computations)
- getVarMatch is expensive just for a fallback and a wrap, bypass it to
  reduce redundant work.
parent 138dfa05
...@@ -15,6 +15,7 @@ ...@@ -15,6 +15,7 @@
# Optimized rendering of global actions (cache) # Optimized rendering of global actions (cache)
from collections import defaultdict
from Products.ERP5Type.Globals import DTMLFile from Products.ERP5Type.Globals import DTMLFile
from Products.ERP5Type import Permissions, _dtmldir from Products.ERP5Type import Permissions, _dtmldir
from Products.DCWorkflow.DCWorkflow import DCWorkflowDefinition, StateChangeInfo, createExprContext from Products.DCWorkflow.DCWorkflow import DCWorkflowDefinition, StateChangeInfo, createExprContext
...@@ -74,7 +75,6 @@ def Guard_checkWithoutRoles(self, sm, wf_def, ob, **kw): ...@@ -74,7 +75,6 @@ def Guard_checkWithoutRoles(self, sm, wf_def, ob, **kw):
because we only want this specific behaviour for worklists (Guards are because we only want this specific behaviour for worklists (Guards are
also used in transitions). also used in transitions).
""" """
u_roles = None
if wf_def.manager_bypass: if wf_def.manager_bypass:
# Possibly bypass. # Possibly bypass.
u_roles = sm.getUser().getRolesInContext(ob) u_roles = sm.getUser().getRolesInContext(ob)
...@@ -103,11 +103,11 @@ def Guard_checkWithoutRoles(self, sm, wf_def, ob, **kw): ...@@ -103,11 +103,11 @@ def Guard_checkWithoutRoles(self, sm, wf_def, ob, **kw):
return 0 return 0
expr = self.expr expr = self.expr
if expr is not None: if expr is not None:
econtext = createExprContext( return expr(createExprContext(StateChangeInfo(
StateChangeInfo(ob, wf_def, kwargs=kw)) ob,
res = expr(econtext) wf_def,
if not res: kwargs=kw,
return 0 )))
return 1 return 1
DCWorkflowDefinition.security = ClassSecurityInfo() DCWorkflowDefinition.security = ClassSecurityInfo()
...@@ -262,81 +262,82 @@ def DCWorkflowDefinition_getWorklistVariableMatchDict(self, info, ...@@ -262,81 +262,82 @@ def DCWorkflowDefinition_getWorklistVariableMatchDict(self, info,
(worklist id as key) and which value is a dict composed of (worklist id as key) and which value is a dict composed of
variable matches. variable matches.
""" """
if not self.worklists: worklist_items = self.worklists.items()
if not worklist_items:
return None return None
portal = self.getPortalObject() portal = self.getPortalObject()
def getPortalTypeListForWorkflow(workflow_id): def getPortalTypeListByWorkflowIdDict():
workflow_tool = portal.portal_workflow workflow_tool = portal.portal_workflow
result = [] result = defaultdict(list)
append = result.append for type_info in workflow_tool._listTypeInfo():
for type_info in workflow_tool._listTypeInfo(): portal_type = type_info.id
portal_type = type_info.id for workflow_id in workflow_tool.getChainFor(portal_type):
if workflow_id in workflow_tool.getChainFor(portal_type): result[workflow_id].append(portal_type)
append(portal_type) return result
return result portal_type_list = CachingMethod(
getPortalTypeListByWorkflowIdDict,
_getPortalTypeListForWorkflow = CachingMethod(getPortalTypeListForWorkflow, id='_getPortalTypeListByWorkflowIdDict',
id='_getPortalTypeListForWorkflow', cache_factory = 'erp5_ui_long') cache_factory='erp5_ui_long',
portal_type_list = _getPortalTypeListForWorkflow(self.id) )()[self.id]
if not portal_type_list: if not portal_type_list:
return None return None
portal_type_set = set(portal_type_list)
variable_match_dict = {} variable_match_dict = {}
security_manager = getSecurityManager() security_manager = getSecurityManager()
workflow_id = self.id workflow_id = self.id
workflow_title = self.title workflow_title = self.title
for worklist_id, worklist_definition in self.worklists.items(): for worklist_id, worklist_definition in worklist_items:
action_box_name = worklist_definition.actbox_name action_box_name = worklist_definition.actbox_name
guard = worklist_definition.guard
if action_box_name: if action_box_name:
variable_match = {} variable_match = {}
for key in worklist_definition.getVarMatchKeys(): for key, var in (worklist_definition.var_matches or {}).iteritems():
var = worklist_definition.getVarMatch(key)
if isinstance(var, Expression): if isinstance(var, Expression):
evaluated_value = var(createExprContext(StateChangeInfo(portal, evaluated_value = var(createExprContext(StateChangeInfo(portal,
self, kwargs=info.__dict__.copy()))) self, kwargs=info.__dict__.copy())))
if isinstance(evaluated_value, (str, int, long)): if isinstance(evaluated_value, (str, int, long)):
evaluated_value = [str(evaluated_value)] evaluated_value = [str(evaluated_value)]
else: else:
if not isinstance(var, tuple):
var = (var, )
evaluated_value = [x % info for x in var] evaluated_value = [x % info for x in var]
variable_match[key] = evaluated_value variable_match[key] = evaluated_value
if 'portal_type' in variable_match and len(variable_match['portal_type']): portal_type_match = variable_match.get('portal_type')
portal_type_intersection = set(variable_match['portal_type'])\ if portal_type_match:
.intersection(portal_type_list)
# in case the current workflow is not associated with portal_types # in case the current workflow is not associated with portal_types
# defined on the worklist, don't display the worklist for this # defined on the worklist, don't display the worklist for this
# portal_type. # portal_type.
variable_match['portal_type'] = list(portal_type_intersection) variable_match['portal_type'] = list(
variable_match.setdefault('portal_type', portal_type_list) portal_type_set.intersection(portal_type_match)
)
if len(variable_match.get('portal_type', [])) == 0: if not variable_match.setdefault('portal_type', portal_type_list):
continue continue
is_permitted_worklist = 0 guard = worklist_definition.guard
if guard is None: if (
is_permitted_worklist = 1 guard is None or
elif (not check_guard) or \ not check_guard or
Guard_checkWithoutRoles(guard, security_manager, self, portal): Guard_checkWithoutRoles(guard, security_manager, self, portal)
is_permitted_worklist = 1 ):
variable_match[SECURITY_PARAMETER_ID] = guard.roles
if is_permitted_worklist:
format_data = TemplateDict() format_data = TemplateDict()
format_data._push(info) format_data._push(info)
variable_match.setdefault(SECURITY_PARAMETER_ID, ()) variable_match.setdefault(SECURITY_PARAMETER_ID, getattr(guard, 'roles', ()))
format_data._push({k: ('&%s:list=' % k).join(v) format_data._push({
for k, v in variable_match.iteritems()}) k: ('&%s:list=' % k).join(v)
variable_match[WORKLIST_METADATA_KEY] = {'format_data': format_data, for k, v in variable_match.iteritems()
'worklist_title': action_box_name, })
'worklist_id': worklist_id, variable_match[WORKLIST_METADATA_KEY] = {
'workflow_title': workflow_title, 'format_data': format_data,
'workflow_id': workflow_id, 'worklist_title': action_box_name,
'action_box_url': worklist_definition.actbox_url, 'worklist_id': worklist_id,
'action_box_category': worklist_definition.actbox_category} 'workflow_title': workflow_title,
'workflow_id': workflow_id,
'action_box_url': worklist_definition.actbox_url,
'action_box_category': worklist_definition.actbox_category,
}
variable_match_dict[worklist_id] = variable_match variable_match_dict[worklist_id] = variable_match
if len(variable_match_dict) == 0: if variable_match_dict:
return None return variable_match_dict
return variable_match_dict return None
DCWorkflowDefinition.security.declarePrivate('getWorklistVariableMatchDict') DCWorkflowDefinition.security.declarePrivate('getWorklistVariableMatchDict')
DCWorkflowDefinition.getWorklistVariableMatchDict = DCWorkflowDefinition_getWorklistVariableMatchDict DCWorkflowDefinition.getWorklistVariableMatchDict = DCWorkflowDefinition_getWorklistVariableMatchDict
......
...@@ -457,162 +457,157 @@ def WorkflowTool_listActions(self, info=None, object=None, src__=False): ...@@ -457,162 +457,157 @@ def WorkflowTool_listActions(self, info=None, object=None, src__=False):
""" """
if object is not None or info is None: if object is not None or info is None:
info = self._getOAI(object) info = self._getOAI(object)
chain = self.getChainFor(info.object)
did = {}
actions = [] actions = []
worklist_dict = {} for wf_id in self.getChainFor(info.object):
for wf_id in chain:
did[wf_id] = None
wf = self.getWorkflowById(wf_id) wf = self.getWorkflowById(wf_id)
if wf is not None: if wf is not None:
a = wf.listObjectActions(info) a = wf.listObjectActions(info)
if a is not None: if a is not None:
actions.extend(a) actions.extend(a)
a = wf.getWorklistVariableMatchDict(info)
if a is not None:
worklist_dict[wf_id] = a
wf_ids = self.getWorkflowIds() portal = self.getPortalObject()
for wf_id in wf_ids: portal_url = portal.portal_url()
if not did.has_key(wf_id): def _getWorklistActionList():
worklist_dict = {}
for wf_id in self.getWorkflowIds():
wf = self.getWorkflowById(wf_id) wf = self.getWorkflowById(wf_id)
if wf is not None: if wf is not None:
a = wf.getWorklistVariableMatchDict(info) a = wf.getWorklistVariableMatchDict(info)
if a is not None: if a is not None:
worklist_dict[wf_id] = a worklist_dict[wf_id] = a
if not worklist_dict:
if worklist_dict: return ()
portal = self.getPortalObject() is_anonymous = portal.portal_membership.isAnonymousUser()
portal_url = portal.portal_url() portal_catalog = portal.portal_catalog
def _getWorklistActionList(): sql_catalog = portal_catalog.getSQLCatalog()
is_anonymous = portal.portal_membership.isAnonymousUser() catalog_security_uid_groups_columns_dict = \
portal_catalog = portal.portal_catalog sql_catalog.getSQLCatalogSecurityUidGroupsColumnsDict()
sql_catalog = portal_catalog.getSQLCatalog() getSecurityUidDictAndRoleColumnDict = \
catalog_security_uid_groups_columns_dict = \ portal_catalog.getSecurityUidDictAndRoleColumnDict
sql_catalog.getSQLCatalogSecurityUidGroupsColumnsDict() search_result_ = getattr(self, "Base_getCountFromWorklistTable", None)
getSecurityUidDictAndRoleColumnDict = \ use_cache = search_result_ is not None
portal_catalog.getSecurityUidDictAndRoleColumnDict if use_cache:
search_result_ = getattr(self, "Base_getCountFromWorklistTable", None) ignored_security_column_id_set = self._getWorklistIgnoredSecurityColumnSet()
use_cache = search_result_ is not None ignored_security_uid_parameter_set = {x
if use_cache: for x, y in catalog_security_uid_groups_columns_dict.iteritems()
ignored_security_column_id_set = self._getWorklistIgnoredSecurityColumnSet() if y in ignored_security_column_id_set
ignored_security_uid_parameter_set = {x }
for x, y in catalog_security_uid_groups_columns_dict.iteritems() _getSecurityUidDictAndRoleColumnDict = getSecurityUidDictAndRoleColumnDict
if y in ignored_security_column_id_set def getSecurityUidDictAndRoleColumnDict(**kw):
} security_uid_dict, role_column_dict, local_role_column_dict = \
_getSecurityUidDictAndRoleColumnDict = getSecurityUidDictAndRoleColumnDict _getSecurityUidDictAndRoleColumnDict(**kw)
def getSecurityUidDictAndRoleColumnDict(**kw): for ignored_security_column_id in ignored_security_column_id_set:
security_uid_dict, role_column_dict, local_role_column_dict = \ role_column_dict.pop(ignored_security_column_id, None)
_getSecurityUidDictAndRoleColumnDict(**kw) local_role_column_dict.pop(ignored_security_column_id, None)
for ignored_security_column_id in ignored_security_column_id_set: for ignored_security_uid_parameter in \
role_column_dict.pop(ignored_security_column_id, None) ignored_security_uid_parameter_set:
local_role_column_dict.pop(ignored_security_column_id, None) security_uid_dict.pop(ignored_security_uid_parameter)
for ignored_security_uid_parameter in \ return security_uid_dict, role_column_dict, local_role_column_dict
ignored_security_uid_parameter_set: count_column_expression = 'sum(`%s`)' % (COUNT_COLUMN_TITLE, )
security_uid_dict.pop(ignored_security_uid_parameter) # Prevent catalog from trying to join
return security_uid_dict, role_column_dict, local_role_column_dict getQuery = SimpleQuery
count_column_expression = 'sum(`%s`)' % (COUNT_COLUMN_TITLE, ) # BBB
# Prevent catalog from trying to join def search_result(select_dict, group_by, query, limit, src__):
getQuery = SimpleQuery select_item_list = []
# BBB for alias, expression in select_dict.iteritems():
def search_result(select_dict, group_by, query, limit, src__): if expression is None:
select_item_list = [] expression = alias
for alias, expression in select_dict.iteritems(): select_item_list.append('%s AS %s' % (expression, alias))
if expression is None: return search_result_(
expression = alias select_expression=','.join(select_item_list),
select_item_list.append('%s AS %s' % (expression, alias)) group_by_expression=','.join(group_by),
return search_result_( query=query,
select_expression=','.join(select_item_list), limit=limit,
group_by_expression=','.join(group_by), src__=src__,
query=query,
limit=limit,
src__=src__,
)
else:
search_result = portal_catalog.unrestrictedSearchResults
count_column_expression = 'count(*)'
# Let catalog join as needed
getQuery = lambda comparison_operator=None, **kw: AutoQuery(
operator=comparison_operator,
**kw
) )
worklist_result_dict = {} else:
# Get a list of dict of WorklistVariableMatchDict grouped by compatible search_result = portal_catalog.unrestrictedSearchResults
# conditions count_column_expression = 'count(*)'
(worklist_list_grouped_by_condition, worklist_metadata) = \ # Let catalog join as needed
groupWorklistListByCondition( getQuery = lambda comparison_operator=None, **kw: AutoQuery(
worklist_dict=worklist_dict, operator=comparison_operator,
sql_catalog=sql_catalog, **kw
getSecurityUidDictAndRoleColumnDict=\ )
getSecurityUidDictAndRoleColumnDict, worklist_result_dict = {}
catalog_security_uid_groups_columns_dict=\ # Get a list of dict of WorklistVariableMatchDict grouped by compatible
catalog_security_uid_groups_columns_dict, # conditions
(worklist_list_grouped_by_condition, worklist_metadata) = \
groupWorklistListByCondition(
worklist_dict=worklist_dict,
sql_catalog=sql_catalog,
getSecurityUidDictAndRoleColumnDict=\
getSecurityUidDictAndRoleColumnDict,
catalog_security_uid_groups_columns_dict=\
catalog_security_uid_groups_columns_dict,
)
if src__:
action_list = []
for grouped_worklist_dict in worklist_list_grouped_by_condition:
# Generate the query for this worklist_list
(total_criterion_id_list, query) = \
getWorklistListQuery(
getQuery=getQuery,
grouped_worklist_dict=grouped_worklist_dict,
) )
if src__: group_by = total_criterion_id_list
action_list = [] assert COUNT_COLUMN_TITLE not in total_criterion_id_list
for grouped_worklist_dict in worklist_list_grouped_by_condition: select_dict = dict.fromkeys(total_criterion_id_list)
# Generate the query for this worklist_list select_dict[COUNT_COLUMN_TITLE] = count_column_expression
(total_criterion_id_list, query) = \ catalog_brain_result = []
getWorklistListQuery( try:
getQuery=getQuery, catalog_brain_result = search_result(
grouped_worklist_dict=grouped_worklist_dict, select_dict=select_dict,
) group_by=group_by,
group_by = total_criterion_id_list query=query,
assert COUNT_COLUMN_TITLE not in total_criterion_id_list limit=None,
select_dict = dict.fromkeys(total_criterion_id_list) src__=src__)
select_dict[COUNT_COLUMN_TITLE] = count_column_expression except Unauthorized:
catalog_brain_result = [] if not is_anonymous:
raise
LOG('WorkflowTool.listActions', WARNING,
'Exception while computing worklists: %s'
% grouped_worklist_dict.keys(),
error=sys.exc_info())
continue
except ProgrammingError, error_value:
# 1146 = table does not exist
if not use_cache or error_value[0] != 1146:
raise
try: try:
catalog_brain_result = search_result( self.Base_zCreateWorklistTable()
select_dict=select_dict,
group_by=group_by,
query=query,
limit=None,
src__=src__)
except Unauthorized:
if not is_anonymous:
raise
LOG('WorkflowTool.listActions', WARNING,
'Exception while computing worklists: %s'
% grouped_worklist_dict.keys(),
error=sys.exc_info())
continue
except ProgrammingError, error_value: except ProgrammingError, error_value:
# 1146 = table does not exist # 1050 = table exists (alarm run just a bit too late)
if not use_cache or error_value[0] != 1146: if error_value[0] != 1050:
raise raise
try: if src__:
self.Base_zCreateWorklistTable() action_list.append(catalog_brain_result)
except ProgrammingError, error_value: else:
# 1050 = table exists (alarm run just a bit too late) grouped_worklist_result = sumCatalogResultByWorklist(
if error_value[0] != 1050: grouped_worklist_dict=grouped_worklist_dict,
raise catalog_result=catalog_brain_result)
if src__: for key, value in grouped_worklist_result.iteritems():
action_list.append(catalog_brain_result) worklist_result_dict[key] = value + worklist_result_dict.get(key, 0)
else: if not src__:
grouped_worklist_result = sumCatalogResultByWorklist( action_list = sorted(
grouped_worklist_dict=grouped_worklist_dict, generateActionList(
catalog_result=catalog_brain_result) worklist_metadata=worklist_metadata,
for key, value in grouped_worklist_result.iteritems(): worklist_result=worklist_result_dict,
worklist_result_dict[key] = value + worklist_result_dict.get(key, 0) portal_url=portal_url),
if not src__: key=lambda x: '/'.join((x['workflow_id'], x['worklist_id'])),
action_list = sorted( )
generateActionList( return action_list
worklist_metadata=worklist_metadata, if src__:
worklist_result=worklist_result_dict, actions = _getWorklistActionList()
portal_url=portal_url), else:
key=lambda x: '/'.join((x['workflow_id'], x['worklist_id'])), actions.extend(CachingMethod(
) _getWorklistActionList,
return action_list id=(
user = _getAuthenticatedUser(self).getIdOrUserName() '_getWorklistActionList',
if src__: _getAuthenticatedUser(self).getIdOrUserName(),
actions = _getWorklistActionList() portal_url,
else: ),
_getWorklistActionList = CachingMethod(_getWorklistActionList, cache_factory = 'erp5_ui_short',
id=('_getWorklistActionList', user, portal_url), )())
cache_factory = 'erp5_ui_short')
actions.extend(_getWorklistActionList())
return actions return actions
WorkflowTool.listActions = WorkflowTool_listActions WorkflowTool.listActions = WorkflowTool_listActions
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment