source: main/waeup.kofa/trunk/src/waeup/kofa/students/payments.py @ 13418

Last change on this file since 13418 was 13398, checked in by Henrik Bettermann, 9 years ago

All kind of school fee payments trigger the same actions. AAUE installment payment is deprecated.

  • Property svn:keywords set to Id
File size: 7.7 KB
RevLine 
[7191]1## $Id: payments.py 13398 2015-11-06 10:59:22Z henrik $
2##
[6635]3## Copyright (C) 2011 Uli Fouquet & Henrik Bettermann
4## This program is free software; you can redistribute it and/or modify
5## it under the terms of the GNU General Public License as published by
6## the Free Software Foundation; either version 2 of the License, or
7## (at your option) any later version.
8##
9## This program is distributed in the hope that it will be useful,
10## but WITHOUT ANY WARRANTY; without even the implied warranty of
11## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12## GNU General Public License for more details.
13##
14## You should have received a copy of the GNU General Public License
15## along with this program; if not, write to the Free Software
16## Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
17##
18"""
[7599]19Student payment components.
[6635]20"""
21import grok
22from zope.component.interfaces import IFactory
[6875]23from zope.interface import implementedBy
[12889]24from zope.schema.interfaces import ConstraintNotSatisfied
25from hurry.workflow.interfaces import IWorkflowInfo
[8420]26from waeup.kofa.interfaces import MessageFactory as _
[7811]27from waeup.kofa.students.interfaces import (
[6877]28    IStudentPaymentsContainer, IStudentNavigation, IStudentOnlinePayment)
[12889]29from waeup.kofa.students.workflow import CLEARED, RETURNING, PAID
[7811]30from waeup.kofa.payments import PaymentsContainer, OnlinePayment
[10030]31from waeup.kofa.payments.interfaces import IPayer
[7811]32from waeup.kofa.utils.helpers import attrs_to_fields
[8420]33from waeup.kofa.accesscodes import create_accesscode
[6635]34
[6860]35class StudentPaymentsContainer(PaymentsContainer):
[6635]36    """This is a container for student payments.
37    """
[6860]38    grok.implements(IStudentPaymentsContainer, IStudentNavigation)
39    grok.provides(IStudentPaymentsContainer)
[6635]40
41    def __init__(self):
[6860]42        super(StudentPaymentsContainer, self).__init__()
[6635]43        return
44
[8736]45    @property
46    def student(self):
[6642]47        return self.__parent__
48
[8735]49    def writeLogMessage(self, view, message):
50        return self.__parent__.writeLogMessage(view, message)
51
[6875]52StudentPaymentsContainer = attrs_to_fields(StudentPaymentsContainer)
53
54class StudentOnlinePayment(OnlinePayment):
55    """This is an online payment.
56    """
[6877]57    grok.implements(IStudentOnlinePayment, IStudentNavigation)
58    grok.provides(IStudentOnlinePayment)
[6875]59
60    def __init__(self):
61        super(StudentOnlinePayment, self).__init__()
62        return
63
[8736]64    @property
65    def student(self):
[8368]66        try:
67            return self.__parent__.__parent__
68        except AttributeError:
69            return None
[6875]70
[8735]71    def writeLogMessage(self, view, message):
72        return self.__parent__.__parent__.writeLogMessage(view, message)
73
[13042]74    def redeemTicket(self):
75        """Either create an appropriate access code or trigger an action
76        directly.
77        """
[8736]78        student = self.student
[8420]79        if self.p_category == 'clearance':
80            # Create CLR access code
81            pin, error = create_accesscode(
82                'CLR',0,self.amount_auth,student.student_id)
83            if error:
[8732]84                return error
[8420]85            self.ac = pin
[13398]86        elif self.p_category.startswith('schoolfee'):
[12889]87            # Bypass activation code creation if next session
88            # can be started directly.
89            if student['studycourse'].next_session_allowed:
90                try:
91                    if student.state == CLEARED:
92                        IWorkflowInfo(student).fireTransition(
93                            'pay_first_school_fee')
94                        return None
95                    elif student.state == RETURNING:
96                        IWorkflowInfo(student).fireTransition(
97                            'pay_school_fee')
98                        return None
99                    elif student.state == PAID:
100                        IWorkflowInfo(student).fireTransition(
101                            'pay_pg_fee')
102                        return None
103                except ConstraintNotSatisfied:
104                    pass
[8420]105            # Create SFE access code
106            pin, error = create_accesscode(
107                'SFE',0,self.amount_auth,student.student_id)
108            if error:
[8732]109                return error
[8420]110            self.ac = pin
111        elif self.p_category == 'bed_allocation':
112            # Create HOS access code
113            pin, error = create_accesscode(
114                'HOS',0,self.amount_auth,student.student_id)
115            if error:
[8732]116                return error
[8420]117            self.ac = pin
[10449]118        elif self.p_category == 'transcript':
119            # Create TSC access code
120            pin, error = create_accesscode(
121                'TSC',0,self.amount_auth,student.student_id)
122            if error:
123                return error
124            self.ac = pin
[8732]125        return None
126
127    def doAfterStudentPayment(self):
128        """Process student after payment was made.
129        """
[9148]130        if self.p_current:
[13042]131            error = self.redeemTicket()
[9148]132            if error is not None:
[11580]133                return 'danger', error, error
[9438]134        log = 'successful %s payment: %s' % (self.p_category, self.p_id)
[8428]135        msg = _('Successful payment')
[11580]136        flashtype = 'success'
137        return flashtype, msg, log
[8420]138
[8453]139    def doAfterStudentPaymentApproval(self):
140        """Process student after payment was approved.
141        """
[9148]142        if self.p_current:
[13042]143            error = self.redeemTicket()
[9148]144            if error is not None:
[11580]145                return 'danger', error, error
[9438]146        log = '%s payment approved: %s' % (self.p_category, self.p_id)
[11583]147        msg = _('Payment approved.')
[11580]148        flashtype = 'success'
149        return flashtype, msg, log
[8453]150
[8422]151    def approveStudentPayment(self):
152        """Approve payment and process student.
153        """
154        if self.p_state == 'paid':
[11580]155            return 'warning', _('This ticket has already been paid.'), None
[8422]156        self.approve()
[8732]157        return self.doAfterStudentPaymentApproval()
[8422]158
159
[9984]160StudentOnlinePayment = attrs_to_fields(
161    StudentOnlinePayment, omit=['display_item'])
[6875]162
[10030]163class Payer(grok.Adapter):
164    """An adapter to publish student data through a simple webservice.
[8703]165    """
166    grok.context(IStudentOnlinePayment)
[10030]167    grok.implements(IPayer)
[8703]168
169    @property
[8708]170    def display_fullname(self):
[10030]171        "Name of  payer"
[8736]172        return self.context.student.display_fullname
[8703]173
[8708]174    @property
175    def id(self):
[10030]176        "Id of payer"
[8736]177        return self.context.student.student_id
[8708]178
179    @property
[9733]180    def matric_number(self):
[10030]181        "Matric number or reg number of payer"
[9506]182        return self.context.student.matric_number
183
184    @property
[9733]185    def reg_number(self):
[10030]186        "Reg number or reg number of payer"
[9733]187        return self.context.student.reg_number
188
189    @property
[8708]190    def faculty(self):
[10030]191        "Faculty of payer"
[8736]192        return self.context.student.faccode
[8708]193
194    @property
195    def department(self):
[10030]196        "Department of payer"
[8736]197        return self.context.student.depcode
[8708]198
[10906]199    @property
200    def email(self):
201        "Email of payer"
202        return self.context.student.email
203
204    @property
205    def phone(self):
206        "Phone number of payer"
207        return self.context.student.phone
208
209    @property
210    def current_mode(self):
211        "Current study mode of payer"
212        return self.context.student.current_mode
213
214    @property
215    def current_level(self):
216        "Current level of payer"
217        return self.context.student.current_level
218
[6875]219# Student online payments must be importable. So we might need a factory.
220class StudentOnlinePaymentFactory(grok.GlobalUtility):
221    """A factory for student online payments.
222    """
223    grok.implements(IFactory)
224    grok.name(u'waeup.StudentOnlinePayment')
225    title = u"Create a new online payment.",
226    description = u"This factory instantiates new online payment instances."
227
228    def __call__(self, *args, **kw):
229        return StudentOnlinePayment()
230
231    def getInterfaces(self):
[7811]232        return implementedBy(StudentOnlinePayment)
Note: See TracBrowser for help on using the repository browser.