source: main/waeup.kofa/trunk/src/waeup/kofa/students/export.py @ 15844

Last change on this file since 15844 was 15792, checked in by Henrik Bettermann, 5 years ago

Add payment status scholarship.

  • Property svn:keywords set to Id
File size: 30.1 KB
RevLine 
[8057]1## $Id: export.py 15792 2019-11-10 20:11:40Z henrik $
2##
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##
[7944]18"""Exporters for student related stuff.
19"""
[9937]20import os
[7944]21import grok
[13156]22from datetime import datetime, timedelta
[9937]23from zope.component import getUtility
[11757]24from waeup.kofa.interfaces import (
25    IExtFileStore, IFileStoreNameChooser, IKofaUtils)
[7944]26from waeup.kofa.interfaces import MessageFactory as _
[9843]27from waeup.kofa.students.catalog import StudentsQuery, CourseTicketsQuery
[8015]28from waeup.kofa.students.interfaces import (
[8371]29    IStudent, IStudentStudyCourse, IStudentStudyLevel, ICourseTicket,
[9427]30    IStudentOnlinePayment, ICSVStudentExporter, IBedTicket)
[9787]31from waeup.kofa.students.vocabularies import study_levels
[7944]32from waeup.kofa.utils.batching import ExporterBase
[11757]33from waeup.kofa.utils.helpers import iface_names, to_timezone
[7944]34
[8400]35
[9787]36def get_students(site, stud_filter=StudentsQuery()):
[8414]37    """Get all students registered in catalog in `site`.
[7944]38    """
[9787]39    return stud_filter.query()
[8414]40
41def get_studycourses(students):
42    """Get studycourses of `students`.
43    """
44    return [x.get('studycourse', None) for x in students
45            if x is not None]
46
47def get_levels(students):
48    """Get all studylevels of `students`.
49    """
50    levels = []
51    for course in get_studycourses(students):
52        for level in course.values():
53            levels.append(level)
54    return levels
55
[9844]56def get_tickets(students, **kw):
57    """Get course tickets of `students`.
[10017]58    If code is passed through, filter course tickets
[14984]59    which belong to this course code and meet level=level
60    and level_session=level_session.
[15546]61    If not, but ct_level, ct_session and ct_semester
[14984]62    are passed through, filter course tickets
[15546]63    which meet level==ct_level, level_session==ct_session
64    and semester==ct_semester.
[8414]65    """
66    tickets = []
[9844]67    code = kw.get('code', None)
[10017]68    level = kw.get('level', None)
[14984]69    session = kw.get('session', None)
70    ct_level = kw.get('ct_level', None)
71    ct_session = kw.get('ct_session', None)
[15546]72    ct_semester = kw.get('ct_semester', None)
[9844]73    if code is None:
[10017]74        for level_obj in get_levels(students):
75            for ticket in level_obj.values():
[14984]76                if ct_level not in ('all', None):
77                    if level_obj.level in (10, 999, None)  \
78                        and int(ct_level) != level_obj.level:
79                        continue
80                    if level_obj.level not in range(
81                        int(ct_level), int(ct_level)+100, 10):
82                        continue
83                if ct_session not in ('all', None) and \
84                    int(ct_session) != level_obj.level_session:
85                    continue
[15546]86                if ct_semester not in ('all', None) and \
87                    int(ct_semester) != ticket.semester:
88                    continue
[9844]89                tickets.append(ticket)
90    else:
[10017]91        for level_obj in get_levels(students):
92            for ticket in level_obj.values():
93                if ticket.code != code:
94                    continue
[14984]95                if level not in ('all', None):
[11483]96                    if level_obj.level in (10, 999, None)  \
97                        and int(level) != level_obj.level:
98                        continue
[14984]99                    if level_obj.level not in range(
100                        int(level), int(level)+100, 10):
[11483]101                        continue
[14984]102                if session not in ('all', None) and \
103                    int(session) != level_obj.level_session:
[10017]104                    continue
105                tickets.append(ticket)
[8414]106    return tickets
107
[13894]108def get_tickets_for_lecturer(students, **kw):
109    """Get course tickets of `students`.
110    Filter course tickets which belong to this course code and
111    which are editable by lecturers.
112    """
113    tickets = []
114    code = kw.get('code', None)
115    level_session = kw.get('session', None)
116    level = kw.get('level', None)
117    for level_obj in get_levels(students):
118        for ticket in level_obj.values():
119            if ticket.code != code:
120                continue
121            if not ticket.editable_by_lecturer:
122                continue
[14984]123            if level not in ('all', None):
[13894]124                if level_obj.level in (10, 999, None)  \
125                    and int(level) != level_obj.level:
126                    continue
[14984]127                if level_obj.level not in range(int(level), int(level)+100, 10):
[13894]128                    continue
[14984]129            if level_session not in ('all', None) and \
[13894]130                int(level_session) != level_obj.level_session:
131                continue
132            tickets.append(ticket)
133    return tickets
134
[14761]135def get_payments(students, p_states=None, **kw):
[11730]136    """Get all payments of `students` within given payment_date period.
[10296]137    """
[11730]138    date_format = '%d/%m/%Y'
[10296]139    payments = []
[15043]140    p_start = kw.get('payments_start')
141    p_end = kw.get('payments_end')
[15042]142    paycat = kw.get('paycat')
[15055]143    paysession = kw.get('paysession')
[15042]144    for student in students:
145        for payment in student.get('payments', {}).values():
[15043]146            if p_start and p_end:
[15042]147                if not payment.payment_date:
148                    continue
[15043]149                payments_start = datetime.strptime(p_start, date_format)
150                payments_end = datetime.strptime(p_end, date_format)
[15042]151                tz = getUtility(IKofaUtils).tzinfo
152                payments_start = tz.localize(payments_start)
153                payments_end = tz.localize(payments_end) + timedelta(days=1)
154                payment_date = to_timezone(payment.payment_date, tz)
155                if payment_date < payments_start or payment_date > payments_end:
156                    continue
157            if p_states and not payment.p_state in p_states:
158                continue
159            if paycat not in ('all', None) and payment.p_category != paycat:
160                continue
[15297]161            if paysession not in ('all', None) \
162                and payment.p_session != int(paysession):
[15055]163                continue
[15042]164            payments.append(payment)
[10296]165    return payments
166
[9427]167def get_bedtickets(students):
168    """Get all bedtickets of `students`.
169    """
170    tickets = []
171    for student in students:
172        for ticket in student.get('accommodation', {}).values():
173            tickets.append(ticket)
174    return tickets
[8414]175
176class StudentExporterBase(ExporterBase):
177    """Exporter for students or related objects.
178    This is a baseclass.
179    """
180    grok.baseclass()
[8411]181    grok.implements(ICSVStudentExporter)
182    grok.provides(ICSVStudentExporter)
[8414]183
[9844]184    def filter_func(self, x, **kw):
[9802]185        return x
[9797]186
[9801]187    def get_filtered(self, site, **kw):
[9843]188        """Get students from a catalog filtered by keywords.
189        students_catalog is the default catalog. The keys must be valid
[9933]190        catalog index names.
[9843]191        Returns a simple empty list, a list with `Student`
192        objects or a catalog result set with `Student`
[9801]193        objects.
194
195        .. seealso:: `waeup.kofa.students.catalog.StudentsCatalog`
196
197        """
[9843]198        # Pass only given keywords to create FilteredCatalogQuery objects.
199        # This way we avoid
[9801]200        # trouble with `None` value ambivalences and queries are also
201        # faster (normally less indexes to ask). Drawback is, that
202        # developers must look into catalog to see what keywords are
203        # valid.
[9845]204        if kw.get('catalog', None) == 'coursetickets':
[9843]205            coursetickets = CourseTicketsQuery(**kw).query()
206            students = []
207            for ticket in coursetickets:
208                students.append(ticket.student)
209            return list(set(students))
[15042]210        # Payments can be filtered by payment date and payment category.
211        # These parameters are not keys of the catalog and must thus be
212        # removed from kw.
[11730]213        try:
214            del kw['payments_start']
215            del kw['payments_end']
[15042]216            del kw['paycat']
[15055]217            del kw['paysession']
[11730]218        except KeyError:
219            pass
[15055]220        # Coursetickets can be filtered by level and session.
[15042]221        # These parameters are not keys of the catalog and must thus be
222        # removed from kw.
[14984]223        try:
224            del kw['ct_level']
225            del kw['ct_session']
[15546]226            del kw['ct_semester']
[14984]227        except KeyError:
228            pass
[9801]229        query = StudentsQuery(**kw)
[9797]230        return query.query()
231
[12518]232    def get_selected(self, site, selected):
233        """Get set of selected students.
234        Returns a simple empty list or a list with `Student`
235        objects.
236        """
237        students = []
238        students_container = site.get('students', {})
239        for id in selected:
240            student = students_container.get(id, None)
[14443]241            if student is None:
242                # try matric number
243                result = list(StudentsQuery(matric_number=id).query())
244                if result:
245                    student = result[0]
246                else:
247                    continue
248            students.append(student)
[12518]249        return students
250
[8414]251    def export(self, values, filepath=None):
252        """Export `values`, an iterable, as CSV file.
253        If `filepath` is ``None``, a raw string with CSV data is returned.
254        """
255        writer, outfile = self.get_csv_writer(filepath)
256        for value in values:
257            self.write_item(value, writer)
258        return self.close_outfile(filepath, outfile)
259
[9797]260    def export_all(self, site, filepath=None):
261        """Export students into filepath as CSV data.
262        If `filepath` is ``None``, a raw string with CSV data is returned.
[9763]263        """
[9802]264        return self.export(self.filter_func(get_students(site)), filepath)
[8414]265
[9797]266    def export_student(self, student, filepath=None):
267        return self.export(self.filter_func([student]), filepath=filepath)
[9734]268
[9802]269    def export_filtered(self, site, filepath=None, **kw):
270        """Export items denoted by `kw`.
271        If `filepath` is ``None``, a raw string with CSV data should
272        be returned.
273        """
274        data = self.get_filtered(site, **kw)
[9844]275        return self.export(self.filter_func(data, **kw), filepath=filepath)
[9802]276
[12518]277    def export_selected(self,site, filepath=None, **kw):
278        """Export data for selected set of students.
279        """
280        selected = kw.get('selected', [])
281        data = self.get_selected(site, selected)
282        return self.export(self.filter_func(data, **kw), filepath=filepath)
[9802]283
[12518]284
[12079]285class StudentExporter(grok.GlobalUtility, StudentExporterBase):
[12861]286    """The Student Exporter first filters the set of students by searching the
287    students catalog. Then it exports student base data of this set of students.
[8414]288    """
[7944]289    grok.name('students')
290
[9936]291    fields = tuple(sorted(iface_names(IStudent))) + (
[9253]292        'password', 'state', 'history', 'certcode', 'is_postgrad',
293        'current_level', 'current_session')
[7944]294    title = _(u'Students')
295
[8493]296    def mangle_value(self, value, name, context=None):
[12861]297        """The mangler prepares the history messages and adds a hash symbol at
298        the end of the phone number to avoid annoying automatic number
299        transformation by Excel or Calc."""
[8493]300        if name == 'history':
301            value = value.messages
[14957]302        if 'phone' in name and value is not None:
[8971]303            # Append hash '#' to phone numbers to circumvent
304            # unwanted excel automatic
[8947]305            value = str('%s#' % value)
[8493]306        return super(
[12079]307            StudentExporter, self).mangle_value(
[8493]308            value, name, context=context)
309
[7944]310
[8414]311class StudentStudyCourseExporter(grok.GlobalUtility, StudentExporterBase):
[12861]312    """The Student Study Course Exporter first filters the set of students
313    by searching the students catalog. Then it exports the data of the current
314    study course container of each student from this set. It does
315    not export their content.
[7994]316    """
317    grok.name('studentstudycourses')
318
[8493]319    fields = tuple(sorted(iface_names(IStudentStudyCourse))) + ('student_id',)
[7994]320    title = _(u'Student Study Courses')
321
[9844]322    def filter_func(self, x, **kw):
[9797]323        return get_studycourses(x)
324
[7994]325    def mangle_value(self, value, name, context=None):
[12861]326        """The mangler determines the certificate code and the student id.
[7994]327        """
328        if name == 'certificate' and value is not None:
329            # XXX: hopefully cert codes are unique site-wide
330            value = value.code
[8493]331        if name == 'student_id' and context is not None:
[8736]332            student = context.student
[8493]333            value = getattr(student, name, None)
[7994]334        return super(
335            StudentStudyCourseExporter, self).mangle_value(
336            value, name, context=context)
337
338
[8414]339class StudentStudyLevelExporter(grok.GlobalUtility, StudentExporterBase):
[12861]340    """The Student Study Level Exporter first filters the set of students
341    by searching the students catalog. Then it exports the data of the student's
[12862]342    study level container data but not their content (course tickets).
[12861]343    The exporter iterates over all objects in the students' ``studycourse``
344    containers.
[8015]345    """
346    grok.name('studentstudylevels')
347
[8493]348    fields = tuple(sorted(iface_names(
[12873]349        IStudentStudyLevel))) + (
[9253]350        'student_id', 'number_of_tickets','certcode')
[8015]351    title = _(u'Student Study Levels')
352
[9844]353    def filter_func(self, x, **kw):
[9802]354        return get_levels(x)
355
[8015]356    def mangle_value(self, value, name, context=None):
[12861]357        """The mangler determines the student id, nothing else.
[8015]358        """
[8493]359        if name == 'student_id' and context is not None:
[8736]360            student = context.student
[8493]361            value = getattr(student, name, None)
[8015]362        return super(
363            StudentStudyLevelExporter, self).mangle_value(
364            value, name, context=context)
365
[8414]366class CourseTicketExporter(grok.GlobalUtility, StudentExporterBase):
[13144]367    """The Course Ticket Exporter exports course tickets. Usually,
[12861]368    the exporter first filters the set of students by searching the
369    students catalog. Then it collects and iterates over all ``studylevel``
370    containers of the filtered student set and finally
371    iterates over all items inside these containers.
372
373    If the course code is passed through, the exporter uses a different
374    catalog. It searches for students in the course tickets catalog and
375    exports those course tickets which belong to the given course code and
[13263]376    also meet level and session passed through at the same time.
[12862]377    This happens if the exporter is called at course level in the academic
[12861]378    section.
[8342]379    """
380    grok.name('coursetickets')
381
[8493]382    fields = tuple(sorted(iface_names(ICourseTicket) +
[11484]383        ['level', 'code', 'level_session'])) + ('student_id',
384        'certcode', 'display_fullname')
[8342]385    title = _(u'Course Tickets')
386
[9844]387    def filter_func(self, x, **kw):
388        return get_tickets(x, **kw)
[9802]389
[8342]390    def mangle_value(self, value, name, context=None):
[12861]391        """The mangler determines the student's id and fullname.
[8342]392        """
393        if context is not None:
[8736]394            student = context.student
[11484]395            if name in ('student_id', 'display_fullname') and student is not None:
[8342]396                value = getattr(student, name, None)
397        return super(
398            CourseTicketExporter, self).mangle_value(
399            value, name, context=context)
400
[13894]401class DataForLecturerExporter(grok.GlobalUtility, StudentExporterBase):
[15049]402    """The Data for Lecturer Exporter searches for students in the course
403    tickets catalog and exports those course tickets which belong to the
404    given course code, meet level and session passed through at the
405    same time, and which are editable by lecturers. This exporter can only
406    be called at course level in the academic section.
[13766]407    """
408    grok.name('lecturer')
[8342]409
[13894]410    fields = ('matric_number', 'student_id', 'display_fullname',
[13766]411              'level', 'code', 'level_session', 'score')
412
413    title = _(u'Data for Lecturer')
414
[13894]415    def filter_func(self, x, **kw):
416        return get_tickets_for_lecturer(x, **kw)
417
[13766]418    def mangle_value(self, value, name, context=None):
419        """The mangler determines the student's id and fullname.
420        """
421        if context is not None:
422            student = context.student
[13885]423            if name in ('matric_number',
424                        'reg_number',
425                        'student_id',
426                        'display_fullname',) and student is not None:
[13766]427                value = getattr(student, name, None)
428        return super(
[13894]429            DataForLecturerExporter, self).mangle_value(
[13766]430            value, name, context=context)
431
[12865]432class StudentPaymentExporter(grok.GlobalUtility, StudentExporterBase):
433    """The Student Payment Exporter first filters the set of students
[12862]434    by searching the students catalog. Then it exports student payment
435    tickets by iterating over the items of the student's ``payments``
[15042]436    container. If the payment period is given, only tickets, which were
[12862]437    paid in payment period, are considered for export.
[8371]438    """
439    grok.name('studentpayments')
440
441    fields = tuple(
[8493]442        sorted(iface_names(
[9984]443            IStudentOnlinePayment, exclude_attribs=False,
[13871]444            omit=['display_item', 'certificate', 'student']))) + (
[13641]445            'student_id','state','current_session')
[8576]446    title = _(u'Student Payments')
[8371]447
[9844]448    def filter_func(self, x, **kw):
[11730]449        return get_payments(x, **kw)
[9802]450
[8371]451    def mangle_value(self, value, name, context=None):
[12862]452        """The mangler determines the student's id, registration
453        state and current session.
[8371]454        """
455        if context is not None:
[8736]456            student = context.student
[10232]457            if name in ['student_id','state',
458                        'current_session'] and student is not None:
[8371]459                value = getattr(student, name, None)
460        return super(
[12865]461            StudentPaymentExporter, self).mangle_value(
[8371]462            value, name, context=context)
463
[12971]464class StudentUnpaidPaymentExporter(StudentPaymentExporter):
465    """The Student Unpaid Payment Exporter works just like the
466    Student Payments Exporter but it exports only unpaid tickets.
467    This exporter is designed for finding and finally purging outdated
468    payment ticket.
469    """
470    grok.name('studentunpaidpayments')
471
472    title = _(u'Student Unpaid Payments')
473
474    def filter_func(self, x, **kw):
[14761]475        return get_payments(x, p_states=('unpaid',) , **kw)
[12971]476
[12865]477class DataForBursaryExporter(StudentPaymentExporter):
[15278]478    """The Data for Bursary Exporter works just like the Student Payment
479    Exporter but it exports much more information about the student. It combines
[12862]480    payment and student data in one table in order to spare postprocessing of
481    two seperate export files. The exporter is primarily used by bursary
[14761]482    officers who have exclusively access to this exporter. The exporter
[15792]483    exports ``paid``, ``waived`` and ``scholarship`` payment tickets.
[10233]484    """
485    grok.name('bursary')
486
[10296]487    def filter_func(self, x, **kw):
[15792]488        return get_payments(x, p_states=('paid', 'waived', 'scholarship'), **kw)
[10296]489
[10233]490    fields = tuple(
491        sorted(iface_names(
492            IStudentOnlinePayment, exclude_attribs=False,
[13871]493            omit=['display_item', 'certificate', 'student']))) + (
[11702]494            'student_id','matric_number','reg_number',
[10236]495            'firstname', 'middlename', 'lastname',
496            'state','current_session',
497            'entry_session', 'entry_mode',
[13943]498            'faccode', 'depcode','certcode')
[10233]499    title = _(u'Payment Data for Bursary')
500
501    def mangle_value(self, value, name, context=None):
[12862]502        """The mangler fetches the student data.
[10233]503        """
504        if context is not None:
505            student = context.student
[10236]506            if name in [
[11702]507                'student_id','matric_number', 'reg_number',
[10236]508                'firstname', 'middlename', 'lastname',
[11702]509                'state', 'current_session',
[10236]510                'entry_session', 'entry_mode',
[11702]511                'faccode', 'depcode', 'certcode'] and student is not None:
[10233]512                value = getattr(student, name, None)
513        return super(
[12865]514            StudentPaymentExporter, self).mangle_value(
[10233]515            value, name, context=context)
516
[15277]517class AccommodationPaymentsExporter(DataForBursaryExporter):
[15278]518    """The Accommodation Payments Exporter works like the Data for Bursary
[15792]519    Exporter above. The exporter exports ``paid``, ``waived`` and ``scholarship``
520    payment tickets with category ``bed_allocation`` or ``hostel_maintenance``.
[15278]521    The exporter is primarily used by accommodation officers who have
522    exclusively access to this exporter.
[15277]523    """
524    grok.name('accommodationpayments')
525
526    def filter_func(self, x, **kw):
527        kw['paycat'] = 'bed_allocation'
[15792]528        payments = get_payments(x, p_states=(
529          'paid', 'waived', 'scholarship'), **kw)
[15277]530        kw['paycat'] = 'hostel_maintenance'
[15792]531        payments += get_payments(x, p_states=(
532          'paid', 'waived', 'scholarship'), **kw)
[15277]533        return payments
534
535    title = _(u'Accommodation Payments')
536
[12865]537class BedTicketExporter(grok.GlobalUtility, StudentExporterBase):
538    """The Bed Ticket Exporter first filters the set of students
[12862]539    by searching the students catalog. Then it exports bed
540    tickets by iterating over the items of the student's ``accommodation``
541    container.
[9427]542    """
543    grok.name('bedtickets')
544
545    fields = tuple(
546        sorted(iface_names(
[9984]547            IBedTicket, exclude_attribs=False,
[13314]548            omit=['display_coordinates', 'maint_payment_made']))) + (
[9984]549            'student_id', 'actual_bed_type')
[9427]550    title = _(u'Bed Tickets')
551
[9844]552    def filter_func(self, x, **kw):
[9802]553        return get_bedtickets(x)
554
[9427]555    def mangle_value(self, value, name, context=None):
[12862]556        """The mangler determines the student id and the type of the bed
557        which has been booked in the ticket.
[9427]558        """
559        if context is not None:
560            student = context.student
561            if name in ['student_id'] and student is not None:
562                value = getattr(student, name, None)
563        if name == 'bed' and value is not None:
564            value = getattr(value, 'bed_id', None)
565        if name == 'actual_bed_type':
[14395]566            value = getattr(getattr(context, 'bed', None), 'bed_type', None)
[9427]567        return super(
[12865]568            BedTicketExporter, self).mangle_value(
[9427]569            value, name, context=context)
570
[15047]571class SchoolFeePaymentsOverviewExporter(StudentExporter):
572    """The School Fee Payments Overview Exporter first filters the set of students
[12862]573    by searching the students catalog. Then it exports some student base data
574    together with the total school fee amount paid in each year over a
575    predefined year range (current year - 9, ... , current year + 1).
[9574]576    """
[15047]577    grok.name('sfpaymentsoverview')
[9574]578
579    curr_year = datetime.now().year
[14596]580    year_range = range(curr_year - 11, curr_year + 1)
[9574]581    year_range_tuple = tuple([str(year) for year in year_range])
[9983]582    fields = ('student_id', 'matric_number', 'display_fullname',
[9574]583        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
[13641]584        'current_level', 'current_session', 'current_mode',
585        'entry_session', 'reg_number'
[9574]586        ) + year_range_tuple
[15047]587    title = _(u'Student School Fee Payments Overview')
[9574]588
589    def mangle_value(self, value, name, context=None):
[12862]590        """The mangler summarizes the school fee amounts made per year. It
591        iterates over all paid school fee payment tickets and
592        adds together the amounts paid in a year. Waived payments
[15792]593        are marked ``waived`` and scholarship payments marked `scholarship`.
[12862]594        """
[9574]595        if name in self.year_range_tuple and context is not None:
[11661]596            value = 0
[9574]597            for ticket in context['payments'].values():
[12568]598                if ticket.p_category == 'schoolfee' and \
[9574]599                    ticket.p_session == int(name):
[12568]600                    if ticket.p_state == 'waived':
601                        value = 'waived'
602                        break
[15792]603                    if ticket.p_state == 'scholarship':
604                        value = 'scholarship'
605                        break
[12568]606                    if ticket.p_state == 'paid':
607                        try:
608                            value += ticket.amount_auth
609                        except TypeError:
610                            pass
[11661]611            if value == 0:
612                value = ''
[14367]613            elif isinstance(value, float):
614                value = round(value, 2)
[9574]615        return super(
[12079]616            StudentExporter, self).mangle_value(
[9734]617            value, name, context=context)
[9744]618
[15051]619class SessionPaymentsOverviewExporter(StudentExporter):
620    """The Session Payments Overview Exporter first filters the set of students
621    by searching the students catalog. Then it exports some student base data
[15060]622    together with the total amount paid in predefined payment categories
623    over the previous three session (referring to current academic session).
624    Sample output:
625
626    header: ``...schoolfee13,schoolfee14,schoolfee15,gown13,gown14,gown15...``
627
628    data: ``...2000.0,,3000.0,,,1000.0,...``
629
[15062]630    This csv data string means that the student paid 2000.0 school fee in 2013
631    and 3000.0 in 2015. S/He furthermore paid 1000.0 for gown rental in 2015.
[15051]632    """
633    grok.name('sessionpaymentsoverview')
634
635    paycats = ('schoolfee', 'clearance', 'gown', 'transcript')
[15060]636    regular_fields = ('student_id', 'matric_number', 'display_fullname',
[15051]637        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
638        'current_level', 'current_session', 'current_mode',
639        'entry_session', 'reg_number'
[15060]640        )
[15051]641    title = _(u'Session Payments Overview')
642
[15060]643    @property
644    def paycatyears(self):
645        cas = grok.getSite()['configuration'].current_academic_session
646        paycatyears = []
647        if cas:
648            year_range = range(cas - 2, cas+1)
649            year_range_tuple = tuple([str(year)[2:] for year in year_range])
650            paycatyears = [
651                cat+year for cat in self.paycats for year in year_range_tuple]
652        return paycatyears
653
654    @property
655    def fields(self):
656        return self.regular_fields + tuple(self.paycatyears)
657
[15051]658    def mangle_value(self, value, name, context=None):
659        """
660        """
[15060]661        amounts = dict()
662        for catyear in self.paycatyears:
663            amounts[catyear] = 0.0
664        if name[:-2] in self.paycats and context is not None:
[15051]665            for ticket in context['payments'].values():
[15060]666                if ticket.p_category == name[:-2]:
[15051]667                    if ticket.p_state in ('waived', 'paid'):
[15060]668                        if str(ticket.p_session)[2:] == name[-2:]:
669                            amounts[name] += ticket.amount_auth
670            if amounts[name] == 0.0:
671                value = ''
672            elif isinstance(amounts[name], float):
673                value = round(amounts[name], 2)
[15051]674        return super(
675            StudentExporter, self).mangle_value(
676            value, name, context=context)
677
[12079]678class StudentStudyLevelsOverviewExporter(StudentExporter):
[12862]679    """The Student Study Levels Overview Exporter first filters the set of
680    students by searching the students catalog. Then it exports some student
681    base data together with the session key of registered levels.
682    Sample output:
683
684    header: ``...100,110,120,200,210,220,300...``
685
686    data: ``...2010,,,2011,2012,,2013...``
687
688    This csv data string means that level 100 was registered in session
689    2010/2011, level 200 in session 2011/2012, level 210 (200 on 1st probation)
690    in session 2012/2013 and level 300 in session 2013/2014.
[9744]691    """
692    grok.name('studylevelsoverview')
693
[9787]694    avail_levels = tuple([str(x) for x in study_levels(None)])
[9744]695
696    fields = ('student_id', ) + (
697        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
[9761]698        'entry_session', 'current_level', 'current_session',
[9787]699        ) + avail_levels
[9744]700    title = _(u'Student Study Levels Overview')
701
702    def mangle_value(self, value, name, context=None):
[12862]703        """The mangler checks if a given level has been registered. It returns
704        the ``level_session`` attribute of the student study level object
705        if the named level exists.
706        """
[9787]707        if name in self.avail_levels and context is not None:
[9744]708            value = ''
709            for level in context['studycourse'].values():
710                if level.level == int(name):
[9761]711                    value = '%s' % level.level_session
[9744]712                    break
713        return super(
[12079]714            StudentExporter, self).mangle_value(
[9744]715            value, name, context=context)
[9936]716
717class ComboCardDataExporter(grok.GlobalUtility, StudentExporterBase):
[12862]718    """Like all other exporters the Combo Card Data Exporter first filters the
719    set of students by searching the students catalog. Then it exports some
720    student base data which are neccessary to print for the Interswitch combo
721    card (identity card for students). The output contains a ``passport_path``
722    column which contains the filesystem path of the passport image file.
723    If no path is given, no passport image file exists.
[9936]724    """
725    grok.name('combocard')
726
727    fields = ('display_fullname',
[9937]728              'student_id','matric_number',
729              'certificate', 'faculty', 'department', 'passport_path')
[9936]730    title = _(u'Combo Card Data')
731
732    def mangle_value(self, value, name, context=None):
[12862]733        """The mangler determines the titles of faculty, department
734        and certificate. It also computes the path of passport image file
735        stored in the filesystem.
736        """
[9936]737        certificate = context['studycourse'].certificate
738        if name == 'certificate' and certificate is not None:
739            value = certificate.title
740        if name == 'department' and certificate is not None:
[10650]741            value = certificate.__parent__.__parent__.longtitle
[9936]742        if name == 'faculty' and certificate is not None:
[10650]743            value = certificate.__parent__.__parent__.__parent__.longtitle
[9937]744        if name == 'passport_path' and certificate is not None:
[12862]745            file_id = IFileStoreNameChooser(context).chooseName(
746                attr='passport.jpg')
[9937]747            os_path = getUtility(IExtFileStore)._pathFromFileID(file_id)
748            if not os.path.exists(os_path):
749                value = None
750            else:
751                value = '/'.join(os_path.split('/')[-4:])
[9936]752        return super(
753            ComboCardDataExporter, self).mangle_value(
754            value, name, context=context)
Note: See TracBrowser for help on using the repository browser.