
git-svn-id: svn://svn.cy55.de/Zope3/src/cybertools/trunk@3080 fd906abe-77d9-0310-91a1-e0d9ade77398
179 lines
6.1 KiB
Python
179 lines
6.1 KiB
Python
#
|
|
# Copyright (c) 2008 Helmut Merz helmutm@cy55.de
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; if not, write to the Free Software
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
#
|
|
|
|
"""
|
|
Planning and recording activities (work items).
|
|
|
|
$Id$
|
|
"""
|
|
|
|
from zope import component
|
|
from zope.component import adapts
|
|
from zope.interface import implementer, implements
|
|
from zope.traversing.api import getParent
|
|
|
|
from cybertools.organize.interfaces import IWorkItem, IWorkItems
|
|
from cybertools.stateful.base import Stateful
|
|
from cybertools.stateful.definition import StatesDefinition
|
|
from cybertools.stateful.definition import State, Transition
|
|
from cybertools.stateful.interfaces import IStatesDefinition
|
|
from cybertools.tracking.btree import Track, getTimeStamp
|
|
from cybertools.tracking.interfaces import ITrackingStorage
|
|
|
|
|
|
@implementer(IStatesDefinition)
|
|
def workItemStates():
|
|
return StatesDefinition('workItemStates',
|
|
State('created', 'created', ('assign', 'cancel',), color='red'),
|
|
State('assigned', 'assigned', ('start', 'finish', 'cancel', 'transfer'),
|
|
color='yellow'),
|
|
State('running', 'running', ('finish',), color='green'),
|
|
State('finished', 'finished', (), color='blue'),
|
|
State('transferred', 'transferred', (), color='grey'),
|
|
State('cancelled', 'cancelled', (), color='grey'),
|
|
Transition('assign', 'assign', 'assigned'),
|
|
Transition('start', 'start', 'running'),
|
|
Transition('finish', 'finish', 'finished'),
|
|
Transition('transfer', 'transfer', 'transferred'),
|
|
Transition('cancel', 'cancel', 'cancelled'),
|
|
initialState='created')
|
|
|
|
|
|
class WorkItem(Stateful):
|
|
|
|
implements(IWorkItem)
|
|
|
|
statesDefinition = 'organize.workItemStates'
|
|
|
|
def getStatesDefinition(self):
|
|
return component.getUtility(IStatesDefinition, name=self.statesDefinition)
|
|
|
|
|
|
class WorkItemTrack(WorkItem, Track):
|
|
""" A work item that may be stored as a track in a tracking storage.
|
|
"""
|
|
|
|
metadata_attributes = Track.metadata_attributes + ('state',)
|
|
index_attributes = metadata_attributes
|
|
typeName = 'WorkItem'
|
|
|
|
initAttributes = set(['party', 'description', 'predecessor',
|
|
'planStart', 'planEnd', 'planDuration', 'planEffort'])
|
|
|
|
def __init__(self, taskId, runId, userName, data={}):
|
|
super(WorkItemTrack, self).__init__(taskId, runId, userName)
|
|
self.state = self.getState() # make initial state persistent
|
|
self.data['creator'] = userName
|
|
self.data['created'] = self.timeStamp
|
|
self.setInitData(**data)
|
|
|
|
def __getattr__(self, attr):
|
|
if attr not in IWorkItem:
|
|
raise AttributeError(attr)
|
|
return self.data.get(attr, None)
|
|
|
|
@property
|
|
def party(self):
|
|
return self.userName
|
|
|
|
def setInitData(self, **kw):
|
|
for k in kw:
|
|
if k not in self.initAttributes:
|
|
raise ValueError("Illegal initial attribute: '%s'." % k)
|
|
party = kw.pop('party', None)
|
|
if party is not None:
|
|
if self.state != 'created':
|
|
raise ValueError("Attribute 'party' may not be set in the state '%s'" %
|
|
self.state)
|
|
else:
|
|
self.userName = party
|
|
indexChanged = True
|
|
self.checkOverwrite(kw)
|
|
updatePlanData = False
|
|
indexChanged = False
|
|
data = self.data
|
|
for k, v in kw.items():
|
|
data[k] = v
|
|
if k.startswith('plan'):
|
|
updatePlanData = True
|
|
if self.planStart is not None and self.planStart != self.timeStamp:
|
|
self.timeStamp = self.planStart
|
|
indexChanged = True
|
|
if updatePlanData and self.planStart:
|
|
data['planEnd'], data['planDuration'], data['planEffort'] = \
|
|
recalcTimes(self.planStart, self.planEnd,
|
|
self.planDuration, self.planEffort)
|
|
if indexChanged:
|
|
self.reindex()
|
|
|
|
def assign(self, party=None):
|
|
self.doTransition('assign')
|
|
self.data['assigned'] = getTimeStamp()
|
|
if party is not None:
|
|
self.userName = party
|
|
self.reindex()
|
|
|
|
def startWork(self, **kw):
|
|
self.checkOverwrite(kw)
|
|
self.doTransition('start')
|
|
start = self.data['start'] = kw.pop('start', None) or getTimeStamp()
|
|
self.timeStamp = start
|
|
self.reindex()
|
|
|
|
def stopWork(self, transition='finish', **kw):
|
|
self.checkOverwrite(kw)
|
|
self.doTransition(transition)
|
|
self.reindex()
|
|
|
|
def reindex(self):
|
|
getParent(self).updateTrack(self, {}) # force reindex
|
|
|
|
def checkOverwrite(self, kw):
|
|
for k, v in kw.items():
|
|
#old = data.get(k)
|
|
old = getattr(self, k, None)
|
|
if old is not None and old != v:
|
|
raise ValueError("Attribute '%s' already set to '%s'." % (k, old))
|
|
|
|
|
|
def recalcTimes(start, end, duration, effort):
|
|
if duration is None and start and end:
|
|
duration = end - start
|
|
if end is None and start and duration:
|
|
end = start + duration
|
|
if effort is None and duration:
|
|
effort = duration
|
|
return end, duration, effort
|
|
|
|
|
|
class WorkItems(object):
|
|
""" A tracking storage adapter managing work items.
|
|
"""
|
|
|
|
implements(IWorkItems)
|
|
adapts(ITrackingStorage)
|
|
|
|
def __init__(self, context):
|
|
self.context = context
|
|
|
|
def __getitem__(self, key):
|
|
return self.context[key]
|
|
|
|
def add(self, task, party, run=0, **kw):
|
|
trackId = self.context.saveUserTrack(task, run, party, kw)
|
|
return self[trackId]
|