source: main/waeup.kofa/trunk/src/waeup/kofa/students/workflow.py @ 15213

Last change on this file since 15213 was 15163, checked in by Henrik Bettermann, 6 years ago

Merge with /main/waeup.kofa/branches/henrik-transcript-workflow:15127-15162

  • Property svn:keywords set to Id
File size: 9.9 KB
RevLine 
[6637]1"""Workflow for students.
2"""
3import grok
[9161]4from datetime import datetime
[7615]5from zope.component import getUtility
[6637]6from hurry.workflow.workflow import Transition, WorkflowState, NullCondition
7from hurry.workflow.interfaces import IWorkflowState, IWorkflowTransitionEvent
[7811]8from waeup.kofa.interfaces import (
[7819]9    IObjectHistory, IKofaWorkflowInfo, IKofaUtils,
[6990]10    CREATED, ADMITTED, CLEARANCE, REQUESTED, CLEARED, PAID, RETURNING,
[15163]11    REGISTERED, VALIDATED, GRADUATED, TRANSREQ, TRANSVAL, TRANSREL,
12    IExtFileStore)
[7811]13from waeup.kofa.interfaces import MessageFactory as _
[7819]14from waeup.kofa.workflow import KofaWorkflow, KofaWorkflowInfo
[7811]15from waeup.kofa.students.interfaces import IStudent, IStudentsUtils
[9161]16from waeup.kofa.utils.helpers import get_current_principal
[6637]17
[7615]18
[7513]19IMPORTABLE_STATES = (ADMITTED, CLEARANCE, REQUESTED, CLEARED, PAID, RETURNING,
[10452]20    REGISTERED, VALIDATED, GRADUATED)
[7513]21
[9028]22FORBIDDEN_POSTGRAD_STATES = (RETURNING, REGISTERED, VALIDATED)
23
[6637]24REGISTRATION_TRANSITIONS = (
25    Transition(
26        transition_id = 'create',
[7677]27        title = _('Create student'),
[6637]28        source = None,
29        condition = NullCondition,
[9022]30        msg = _('Record created'),
[6637]31        destination = CREATED),
32
33    Transition(
34        transition_id = 'admit',
[7670]35        title = _('Admit student'),
[9022]36        msg = _('Admitted'),
[6637]37        source = CREATED,
38        destination = ADMITTED),
39
40    Transition(
41        transition_id = 'reset1',
[7677]42        title = _('Reset student'),
[7996]43        msg = _('Reset to initial state'),
[6637]44        source = ADMITTED,
45        destination = CREATED),
[6720]46
47    Transition(
48        transition_id = 'start_clearance',
[7677]49        title = _('Start clearance'),
50        msg = _('Clearance started'),
[6720]51        source = ADMITTED,
52        destination = CLEARANCE),
53
54    Transition(
55        transition_id = 'reset2',
[7677]56        title = _('Reset to admitted'),
[7996]57        msg = _("Reset to 'admitted'"),
[6720]58        source = CLEARANCE,
59        destination = ADMITTED),
60
61    Transition(
62        transition_id = 'request_clearance',
[7677]63        title = _('Request clearance'),
64        msg = _('Clearance requested'),
[6720]65        source = CLEARANCE,
66        destination = REQUESTED),
67
68    Transition(
69        transition_id = 'reset3',
[9022]70        title = _('Reset to clearance started'),
71        msg = _("Reset to 'clearance started'"),
[6720]72        source = REQUESTED,
73        destination = CLEARANCE),
74
75    Transition(
76        transition_id = 'clear',
[7677]77        title = _('Clear student'),
78        msg = _('Cleared'),
[6720]79        source = REQUESTED,
80        destination = CLEARED),
81
82    Transition(
83        transition_id = 'reset4',
[9022]84        title = _('Reset to clearance started'),
85        msg = _("Reset to 'clearance started'"),
[6720]86        source = CLEARED,
87        destination = CLEARANCE),
[6742]88
89    Transition(
90        transition_id = 'pay_first_school_fee',
[7677]91        title = _('Pay school fee'),
[9022]92        msg = _('First school fee payment made'),
[6742]93        source = CLEARED,
94        destination = PAID),
95
96    Transition(
[8434]97        transition_id = 'approve_first_school_fee',
98        title = _('Approve payment'),
[9022]99        msg = _('First school fee payment approved'),
[8434]100        source = CLEARED,
101        destination = PAID),
102
103    Transition(
[6742]104        transition_id = 'reset5',
[7677]105        title = _('Reset to cleared'),
[7996]106        msg = _("Reset to 'cleared'"),
[6742]107        source = PAID,
108        destination = CLEARED),
109
110    Transition(
111        transition_id = 'pay_school_fee',
[7677]112        title = _('Pay school fee'),
[9022]113        msg = _('School fee payment made'),
[6742]114        source = RETURNING,
115        destination = PAID),
116
117    Transition(
[8471]118        transition_id = 'pay_pg_fee',
[9022]119        title = _('Pay PG school fee'),
120        msg = _('PG school fee payment made'),
[8471]121        source = PAID,
122        destination = PAID),
123
124    Transition(
[8434]125        transition_id = 'approve_school_fee',
[9022]126        title = _('Approve school fee payment'),
[8434]127        msg = _('School fee payment approved'),
128        source = RETURNING,
129        destination = PAID),
130
131    Transition(
[8471]132        transition_id = 'approve_pg_fee',
[9022]133        title = _('Approve PG school fee payment'),
134        msg = _('PG school fee payment approved'),
[8471]135        source = PAID,
136        destination = PAID),
137
138    Transition(
[6742]139        transition_id = 'reset6',
[7677]140        title = _('Reset to returning'),
[7996]141        msg = _("Reset to 'returning'"),
[6742]142        source = PAID,
143        destination = RETURNING),
[6801]144
145    Transition(
146        transition_id = 'register_courses',
[7677]147        title = _('Register courses'),
148        msg = _('Courses registered'),
[6801]149        source = PAID,
150        destination = REGISTERED),
151
152    Transition(
153        transition_id = 'reset7',
[13610]154        title = _('Unregister courses'),
155        msg = _("Courses unregistered"),
[6801]156        source = REGISTERED,
157        destination = PAID),
158
159    Transition(
160        transition_id = 'validate_courses',
[7677]161        title = _('Validate courses'),
162        msg = _('Courses validated'),
[6801]163        source = REGISTERED,
164        destination = VALIDATED),
165
166    Transition(
[9284]167        transition_id = 'bypass_validation',
168        title = _('Return and bypass validation'),
[9295]169        msg = _("Course validation bypassed"),
[9284]170        source = REGISTERED,
171        destination = RETURNING),
172
173    Transition(
[6801]174        transition_id = 'reset8',
[9022]175        title = _('Reset to school fee paid'),
176        msg = _("Reset to 'school fee paid'"),
[6801]177        source = VALIDATED,
178        destination = PAID),
179
180    Transition(
181        transition_id = 'return',
[7677]182        title = _('Return'),
[9284]183        msg = _("Returned"),
[6801]184        source = VALIDATED,
185        destination = RETURNING),
186
187    Transition(
188        transition_id = 'reset9',
[9022]189        title = _('Reset to courses validated'),
190        msg = _("Reset to 'courses validated'"),
[6801]191        source = RETURNING,
192        destination = VALIDATED),
[10446]193
194    # There is no transition to graduated.
195
196    Transition(
197        transition_id = 'request_transcript',
198        title = _('Request transript'),
199        msg = _("Transcript requested"),
200        source = GRADUATED,
[15163]201        destination = TRANSREQ),
[10446]202
203    Transition(
[15163]204        transition_id = 'reset10',
205        title = _('Reject transcript request'),
206        msg = _("Transcript request rejected"),
207        source = TRANSREQ,
[10446]208        destination = GRADUATED),
[15163]209
210    Transition(
211        transition_id = 'validate_transcript',
212        title = _('Validate transcript'),
213        msg = _("Transcript validated"),
214        source = TRANSREQ,
215        destination = TRANSVAL),
216
217    Transition(
218        transition_id = 'release_transcript',
219        title = _('Release transcript'),
220        msg = _("Transcript released"),
221        source = TRANSVAL,
222        destination = TRANSREL),
223
224    Transition(
225        transition_id = 'reset11',
226        title = _('Reset to graduated'),
227        msg = _("Transcript process reset"),
228        source = TRANSREL,
229        destination = GRADUATED),
[6637]230    )
231
[10446]232
[8309]233IMPORTABLE_TRANSITIONS = [i.transition_id for i in REGISTRATION_TRANSITIONS]
234
[9028]235FORBIDDEN_POSTGRAD_TRANS = ['reset6', 'register_courses']
[6720]236
[7819]237registration_workflow = KofaWorkflow(REGISTRATION_TRANSITIONS)
[6637]238
239class RegistrationWorkflowState(WorkflowState, grok.Adapter):
240    """An adapter to adapt Student objects to workflow states.
241    """
242    grok.context(IStudent)
243    grok.provides(IWorkflowState)
244
245    state_key = 'wf.registration.state'
246    state_id = 'wf.registration.id'
247
[7819]248class RegistrationWorkflowInfo(KofaWorkflowInfo, grok.Adapter):
[6637]249    """Adapter to adapt Student objects to workflow info objects.
250    """
251    grok.context(IStudent)
[7819]252    grok.provides(IKofaWorkflowInfo)
[6637]253
254    def __init__(self, context):
255        self.context = context
256        self.wf = registration_workflow
257
258@grok.subscribe(IStudent, IWorkflowTransitionEvent)
259def handle_student_transition_event(obj, event):
[6644]260    """Append message to student history and log file when transition happened.
[7133]261
[7681]262    Lock and unlock clearance form.
[9005]263    Trigger actions after school fee payment.
[6637]264    """
[7679]265
266    msg = event.transition.user_data['msg']
[6637]267    history = IObjectHistory(obj)
268    history.addMessage(msg)
[6742]269    # School fee payment of returning students triggers the change of
270    # current session, current level, and current verdict
[8434]271    if event.transition.transition_id in (
272        'pay_school_fee', 'approve_school_fee'):
[7615]273        getUtility(IStudentsUtils).setReturningData(obj)
[8471]274    elif event.transition.transition_id in (
275        'pay_pg_fee', 'approve_pg_fee'):
276        new_session = obj['studycourse'].current_session + 1
277        obj['studycourse'].current_session = new_session
[9161]278    elif event.transition.transition_id == 'validate_courses':
279        current_level = obj['studycourse'].current_level
280        level_object = obj['studycourse'].get(str(current_level), None)
281        if level_object is not None:
282            user = get_current_principal()
283            if user is None:
284                usertitle = 'system'
285            else:
286                usertitle = getattr(user, 'public_name', None)
287                if not usertitle:
288                    usertitle = user.title
289            level_object.validated_by = usertitle
290            level_object.validation_date = datetime.utcnow()
[9484]291    elif event.transition.transition_id == 'clear':
[9486]292        obj.officer_comment = None
[9161]293    elif event.transition.transition_id == 'reset8':
294        current_level = obj['studycourse'].current_level
295        level_object = obj['studycourse'].get(str(current_level), None)
296        if level_object is not None:
297            level_object.validated_by = None
298            level_object.validation_date = None
[15163]299    elif event.transition.transition_id == 'reset11':
300        transcript_file = getUtility(IExtFileStore).getFileByContext(
301            obj, attr='final_transcript')
302        if transcript_file:
303            getUtility(IExtFileStore).deleteFileByContext(
304                obj, attr='final_transcript')
305        obj['studycourse'].transcript_comment = None
306        obj['studycourse'].transcript_signees = None
[7652]307    # In some tests we don't have a students container
[6644]308    try:
309        students_container = grok.getSite()['students']
[7652]310        students_container.logger.info('%s - %s' % (obj.student_id,msg))
[6644]311    except (TypeError, AttributeError):
312        pass
[6637]313    return
Note: See TracBrowser for help on using the repository browser.