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

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

Add maint_payment_made method to BedTicket? class.

  • Property svn:keywords set to Id
File size: 23.4 KB
RevLine 
[8057]1## $Id: export.py 13314 2015-10-13 17:29:07Z 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
59    which belong to this course code and meets level
60    and level_session.
[8414]61    """
62    tickets = []
[9844]63    code = kw.get('code', None)
[10017]64    level = kw.get('level', None)
[13253]65    level_session = kw.get('session', None)
[9844]66    if code is None:
[10017]67        for level_obj in get_levels(students):
68            for ticket in level_obj.values():
[9844]69                tickets.append(ticket)
70    else:
[10017]71        for level_obj in get_levels(students):
72            for ticket in level_obj.values():
73                if ticket.code != code:
74                    continue
[11483]75                if level is not None:
76                    level = int(level)
77                    if level_obj.level in (10, 999, None)  \
78                        and int(level) != level_obj.level:
79                        continue
80                    if level_obj.level not in range(level, level+100, 10):
81                        continue
[10017]82                if level_session is not None and \
83                    int(level_session) != level_obj.level_session:
84                    continue
85                tickets.append(ticket)
[8414]86    return tickets
87
[12971]88def get_payments(students, p_state=None, **kw):
[11730]89    """Get all payments of `students` within given payment_date period.
[10296]90    """
[11730]91    date_format = '%d/%m/%Y'
[10296]92    payments = []
[11730]93    payments_start = kw.get('payments_start')
94    payments_end = kw.get('payments_end')
95    if payments_start and payments_end:
96        # Payment period given
97        payments_start = datetime.strptime(payments_start, date_format)
98        payments_end = datetime.strptime(payments_end, date_format)
[11757]99        tz = getUtility(IKofaUtils).tzinfo
100        payments_start = tz.localize(payments_start)
[13156]101        payments_end = tz.localize(payments_end) + timedelta(days=1)
[12971]102        if p_state:
103            # Only paid or unpaid tickets in payment period are considered
[11730]104            for student in students:
105                for payment in student.get('payments', {}).values():
[12971]106                    if payment.payment_date and payment.p_state == p_state:
[11757]107                        payment_date = to_timezone(payment.payment_date, tz)
108                        if payment_date > payments_start and \
109                            payment_date < payments_end:
110                            payments.append(payment)
[11730]111        else:
112            # All tickets in payment period are considered
113            for student in students:
114                for payment in student.get('payments', {}).values():
[11757]115                    if payment.payment_date:
116                        payment_date = to_timezone(payment.payment_date, tz)
117                        if payment_date > payments_start and \
118                            payment_date < payments_end:
119                            payments.append(payment)
[11730]120    else:
121        # Payment period not given
[12971]122        if p_state:
[11730]123            # Only paid tickets are considered
124            for student in students:
125                for payment in student.get('payments', {}).values():
[12971]126                    if payment.p_state == p_state:
[11730]127                        payments.append(payment)
128        else:
129            # All tickets are considered
130            for student in students:
131                for payment in student.get('payments', {}).values():
132                    payments.append(payment)
[10296]133    return payments
134
[9427]135def get_bedtickets(students):
136    """Get all bedtickets of `students`.
137    """
138    tickets = []
139    for student in students:
140        for ticket in student.get('accommodation', {}).values():
141            tickets.append(ticket)
142    return tickets
[8414]143
144class StudentExporterBase(ExporterBase):
145    """Exporter for students or related objects.
146    This is a baseclass.
147    """
148    grok.baseclass()
[8411]149    grok.implements(ICSVStudentExporter)
150    grok.provides(ICSVStudentExporter)
[8414]151
[9844]152    def filter_func(self, x, **kw):
[9802]153        return x
[9797]154
[9801]155    def get_filtered(self, site, **kw):
[9843]156        """Get students from a catalog filtered by keywords.
157        students_catalog is the default catalog. The keys must be valid
[9933]158        catalog index names.
[9843]159        Returns a simple empty list, a list with `Student`
160        objects or a catalog result set with `Student`
[9801]161        objects.
162
163        .. seealso:: `waeup.kofa.students.catalog.StudentsCatalog`
164
165        """
[9843]166        # Pass only given keywords to create FilteredCatalogQuery objects.
167        # This way we avoid
[9801]168        # trouble with `None` value ambivalences and queries are also
169        # faster (normally less indexes to ask). Drawback is, that
170        # developers must look into catalog to see what keywords are
171        # valid.
[9845]172        if kw.get('catalog', None) == 'coursetickets':
[9843]173            coursetickets = CourseTicketsQuery(**kw).query()
174            students = []
175            for ticket in coursetickets:
176                students.append(ticket.student)
177            return list(set(students))
[11730]178        # Payments can be filtered by payment_date. The period boundaries
179        # are not keys of the catalog and must thus be removed from kw.
180        try:
181            del kw['payments_start']
182            del kw['payments_end']
183        except KeyError:
184            pass
[9801]185        query = StudentsQuery(**kw)
[9797]186        return query.query()
187
[12518]188    def get_selected(self, site, selected):
189        """Get set of selected students.
190        Returns a simple empty list or a list with `Student`
191        objects.
192        """
193        students = []
194        students_container = site.get('students', {})
195        for id in selected:
196            student = students_container.get(id, None)
197            if student:
198                students.append(student)
199        return students
200
[8414]201    def export(self, values, filepath=None):
202        """Export `values`, an iterable, as CSV file.
203        If `filepath` is ``None``, a raw string with CSV data is returned.
204        """
205        writer, outfile = self.get_csv_writer(filepath)
206        for value in values:
207            self.write_item(value, writer)
208        return self.close_outfile(filepath, outfile)
209
[9797]210    def export_all(self, site, filepath=None):
211        """Export students into filepath as CSV data.
212        If `filepath` is ``None``, a raw string with CSV data is returned.
[9763]213        """
[9802]214        return self.export(self.filter_func(get_students(site)), filepath)
[8414]215
[9797]216    def export_student(self, student, filepath=None):
217        return self.export(self.filter_func([student]), filepath=filepath)
[9734]218
[9802]219    def export_filtered(self, site, filepath=None, **kw):
220        """Export items denoted by `kw`.
221        If `filepath` is ``None``, a raw string with CSV data should
222        be returned.
223        """
224        data = self.get_filtered(site, **kw)
[9844]225        return self.export(self.filter_func(data, **kw), filepath=filepath)
[9802]226
[12518]227    def export_selected(self,site, filepath=None, **kw):
228        """Export data for selected set of students.
229        """
230        selected = kw.get('selected', [])
231        data = self.get_selected(site, selected)
232        return self.export(self.filter_func(data, **kw), filepath=filepath)
[9802]233
[12518]234
[12079]235class StudentExporter(grok.GlobalUtility, StudentExporterBase):
[12861]236    """The Student Exporter first filters the set of students by searching the
237    students catalog. Then it exports student base data of this set of students.
[8414]238    """
[7944]239    grok.name('students')
240
[9936]241    fields = tuple(sorted(iface_names(IStudent))) + (
[9253]242        'password', 'state', 'history', 'certcode', 'is_postgrad',
243        'current_level', 'current_session')
[7944]244    title = _(u'Students')
245
[8493]246    def mangle_value(self, value, name, context=None):
[12861]247        """The mangler prepares the history messages and adds a hash symbol at
248        the end of the phone number to avoid annoying automatic number
249        transformation by Excel or Calc."""
[8493]250        if name == 'history':
251            value = value.messages
[8971]252        if name == 'phone' and value is not None:
253            # Append hash '#' to phone numbers to circumvent
254            # unwanted excel automatic
[8947]255            value = str('%s#' % value)
[8493]256        return super(
[12079]257            StudentExporter, self).mangle_value(
[8493]258            value, name, context=context)
259
[7944]260
[8414]261class StudentStudyCourseExporter(grok.GlobalUtility, StudentExporterBase):
[12861]262    """The Student Study Course Exporter first filters the set of students
263    by searching the students catalog. Then it exports the data of the current
264    study course container of each student from this set. It does
265    not export their content.
[7994]266    """
267    grok.name('studentstudycourses')
268
[8493]269    fields = tuple(sorted(iface_names(IStudentStudyCourse))) + ('student_id',)
[7994]270    title = _(u'Student Study Courses')
271
[9844]272    def filter_func(self, x, **kw):
[9797]273        return get_studycourses(x)
274
[7994]275    def mangle_value(self, value, name, context=None):
[12861]276        """The mangler determines the certificate code and the student id.
[7994]277        """
278        if name == 'certificate' and value is not None:
279            # XXX: hopefully cert codes are unique site-wide
280            value = value.code
[8493]281        if name == 'student_id' and context is not None:
[8736]282            student = context.student
[8493]283            value = getattr(student, name, None)
[7994]284        return super(
285            StudentStudyCourseExporter, self).mangle_value(
286            value, name, context=context)
287
288
[8414]289class StudentStudyLevelExporter(grok.GlobalUtility, StudentExporterBase):
[12861]290    """The Student Study Level Exporter first filters the set of students
291    by searching the students catalog. Then it exports the data of the student's
[12862]292    study level container data but not their content (course tickets).
[12861]293    The exporter iterates over all objects in the students' ``studycourse``
294    containers.
[8015]295    """
296    grok.name('studentstudylevels')
297
[8493]298    fields = tuple(sorted(iface_names(
[12873]299        IStudentStudyLevel))) + (
[9253]300        'student_id', 'number_of_tickets','certcode')
[8015]301    title = _(u'Student Study Levels')
302
[9844]303    def filter_func(self, x, **kw):
[9802]304        return get_levels(x)
305
[8015]306    def mangle_value(self, value, name, context=None):
[12861]307        """The mangler determines the student id, nothing else.
[8015]308        """
[8493]309        if name == 'student_id' and context is not None:
[8736]310            student = context.student
[8493]311            value = getattr(student, name, None)
[8015]312        return super(
313            StudentStudyLevelExporter, self).mangle_value(
314            value, name, context=context)
315
[8414]316class CourseTicketExporter(grok.GlobalUtility, StudentExporterBase):
[13144]317    """The Course Ticket Exporter exports course tickets. Usually,
[12861]318    the exporter first filters the set of students by searching the
319    students catalog. Then it collects and iterates over all ``studylevel``
320    containers of the filtered student set and finally
321    iterates over all items inside these containers.
322
323    If the course code is passed through, the exporter uses a different
324    catalog. It searches for students in the course tickets catalog and
325    exports those course tickets which belong to the given course code and
[13263]326    also meet level and session passed through at the same time.
[12862]327    This happens if the exporter is called at course level in the academic
[12861]328    section.
[8342]329    """
330    grok.name('coursetickets')
331
[8493]332    fields = tuple(sorted(iface_names(ICourseTicket) +
[11484]333        ['level', 'code', 'level_session'])) + ('student_id',
334        'certcode', 'display_fullname')
[8342]335    title = _(u'Course Tickets')
336
[9844]337    def filter_func(self, x, **kw):
338        return get_tickets(x, **kw)
[9802]339
[8342]340    def mangle_value(self, value, name, context=None):
[12861]341        """The mangler determines the student's id and fullname.
[8342]342        """
343        if context is not None:
[8736]344            student = context.student
[11484]345            if name in ('student_id', 'display_fullname') and student is not None:
[8342]346                value = getattr(student, name, None)
347        return super(
348            CourseTicketExporter, self).mangle_value(
349            value, name, context=context)
350
351
[12865]352class StudentPaymentExporter(grok.GlobalUtility, StudentExporterBase):
353    """The Student Payment Exporter first filters the set of students
[12862]354    by searching the students catalog. Then it exports student payment
355    tickets by iterating over the items of the student's ``payments``
356    container. If the payment period is given only tickets, which were
357    paid in payment period, are considered for export.
[8371]358    """
359    grok.name('studentpayments')
360
361    fields = tuple(
[8493]362        sorted(iface_names(
[9984]363            IStudentOnlinePayment, exclude_attribs=False,
364            omit=['display_item']))) + (
[10232]365            'student_id','state','current_session')
[8576]366    title = _(u'Student Payments')
[8371]367
[9844]368    def filter_func(self, x, **kw):
[11730]369        return get_payments(x, **kw)
[9802]370
[8371]371    def mangle_value(self, value, name, context=None):
[12862]372        """The mangler determines the student's id, registration
373        state and current session.
[8371]374        """
375        if context is not None:
[8736]376            student = context.student
[10232]377            if name in ['student_id','state',
378                        'current_session'] and student is not None:
[8371]379                value = getattr(student, name, None)
380        return super(
[12865]381            StudentPaymentExporter, self).mangle_value(
[8371]382            value, name, context=context)
383
[12971]384class StudentUnpaidPaymentExporter(StudentPaymentExporter):
385    """The Student Unpaid Payment Exporter works just like the
386    Student Payments Exporter but it exports only unpaid tickets.
387    This exporter is designed for finding and finally purging outdated
388    payment ticket.
389    """
390    grok.name('studentunpaidpayments')
391
392    title = _(u'Student Unpaid Payments')
393
394    def filter_func(self, x, **kw):
395        return get_payments(x, p_state='unpaid', **kw)
396
[12865]397class DataForBursaryExporter(StudentPaymentExporter):
[12862]398    """The DataForBursary Exporter works just like the Student Payments Exporter
399    but it exports much more information about the student. It combines
400    payment and student data in one table in order to spare postprocessing of
401    two seperate export files. The exporter is primarily used by bursary
402    officers who have exclusively access to this exporter.
[10233]403    """
404    grok.name('bursary')
405
[10296]406    def filter_func(self, x, **kw):
[12971]407        return get_payments(x, p_state='paid', **kw)
[10296]408
[10233]409    fields = tuple(
410        sorted(iface_names(
411            IStudentOnlinePayment, exclude_attribs=False,
412            omit=['display_item']))) + (
[11702]413            'student_id','matric_number','reg_number',
[10236]414            'firstname', 'middlename', 'lastname',
415            'state','current_session',
416            'entry_session', 'entry_mode',
[11702]417            'faccode', 'depcode','certcode')
[10233]418    title = _(u'Payment Data for Bursary')
419
420    def mangle_value(self, value, name, context=None):
[12862]421        """The mangler fetches the student data.
[10233]422        """
423        if context is not None:
424            student = context.student
[10236]425            if name in [
[11702]426                'student_id','matric_number', 'reg_number',
[10236]427                'firstname', 'middlename', 'lastname',
[11702]428                'state', 'current_session',
[10236]429                'entry_session', 'entry_mode',
[11702]430                'faccode', 'depcode', 'certcode'] and student is not None:
[10233]431                value = getattr(student, name, None)
432        return super(
[12865]433            StudentPaymentExporter, self).mangle_value(
[10233]434            value, name, context=context)
435
[12865]436class BedTicketExporter(grok.GlobalUtility, StudentExporterBase):
437    """The Bed Ticket Exporter first filters the set of students
[12862]438    by searching the students catalog. Then it exports bed
439    tickets by iterating over the items of the student's ``accommodation``
440    container.
[9427]441    """
442    grok.name('bedtickets')
443
444    fields = tuple(
445        sorted(iface_names(
[9984]446            IBedTicket, exclude_attribs=False,
[13314]447            omit=['display_coordinates', 'maint_payment_made']))) + (
[9984]448            'student_id', 'actual_bed_type')
[9427]449    title = _(u'Bed Tickets')
450
[9844]451    def filter_func(self, x, **kw):
[9802]452        return get_bedtickets(x)
453
[9427]454    def mangle_value(self, value, name, context=None):
[12862]455        """The mangler determines the student id and the type of the bed
456        which has been booked in the ticket.
[9427]457        """
458        if context is not None:
459            student = context.student
460            if name in ['student_id'] and student is not None:
461                value = getattr(student, name, None)
462        if name == 'bed' and value is not None:
463            value = getattr(value, 'bed_id', None)
464        if name == 'actual_bed_type':
465            value = getattr(getattr(context, 'bed', None), 'bed_type')
466        return super(
[12865]467            BedTicketExporter, self).mangle_value(
[9427]468            value, name, context=context)
469
[12079]470class StudentPaymentsOverviewExporter(StudentExporter):
[12862]471    """The Student Payments Overview Exporter first filters the set of students
472    by searching the students catalog. Then it exports some student base data
473    together with the total school fee amount paid in each year over a
474    predefined year range (current year - 9, ... , current year + 1).
[9574]475    """
476    grok.name('paymentsoverview')
477
478    curr_year = datetime.now().year
479    year_range = range(curr_year - 9, curr_year + 1)
480    year_range_tuple = tuple([str(year) for year in year_range])
481
[9983]482    fields = ('student_id', 'matric_number', 'display_fullname',
[9574]483        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
[13281]484        'current_level', 'current_session', 'current_mode', 'entry_session'
[9574]485        ) + year_range_tuple
486    title = _(u'Student Payments Overview')
487
488    def mangle_value(self, value, name, context=None):
[12862]489        """The mangler summarizes the school fee amounts made per year. It
490        iterates over all paid school fee payment tickets and
491        adds together the amounts paid in a year. Waived payments
492        are marked ``waived``.
493        """
[9574]494        if name in self.year_range_tuple and context is not None:
[11661]495            value = 0
[9574]496            for ticket in context['payments'].values():
[12568]497                if ticket.p_category == 'schoolfee' and \
[9574]498                    ticket.p_session == int(name):
[12568]499                    if ticket.p_state == 'waived':
500                        value = 'waived'
501                        break
502                    if ticket.p_state == 'paid':
503                        try:
504                            value += ticket.amount_auth
505                        except TypeError:
506                            pass
[11661]507            if value == 0:
508                value = ''
[9574]509        return super(
[12079]510            StudentExporter, self).mangle_value(
[9734]511            value, name, context=context)
[9744]512
[12079]513class StudentStudyLevelsOverviewExporter(StudentExporter):
[12862]514    """The Student Study Levels Overview Exporter first filters the set of
515    students by searching the students catalog. Then it exports some student
516    base data together with the session key of registered levels.
517    Sample output:
518
519    header: ``...100,110,120,200,210,220,300...``
520
521    data: ``...2010,,,2011,2012,,2013...``
522
523    This csv data string means that level 100 was registered in session
524    2010/2011, level 200 in session 2011/2012, level 210 (200 on 1st probation)
525    in session 2012/2013 and level 300 in session 2013/2014.
[9744]526    """
527    grok.name('studylevelsoverview')
528
[9787]529    avail_levels = tuple([str(x) for x in study_levels(None)])
[9744]530
531    fields = ('student_id', ) + (
532        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
[9761]533        'entry_session', 'current_level', 'current_session',
[9787]534        ) + avail_levels
[9744]535    title = _(u'Student Study Levels Overview')
536
537    def mangle_value(self, value, name, context=None):
[12862]538        """The mangler checks if a given level has been registered. It returns
539        the ``level_session`` attribute of the student study level object
540        if the named level exists.
541        """
[9787]542        if name in self.avail_levels and context is not None:
[9744]543            value = ''
544            for level in context['studycourse'].values():
545                if level.level == int(name):
[9761]546                    value = '%s' % level.level_session
[9744]547                    break
548        return super(
[12079]549            StudentExporter, self).mangle_value(
[9744]550            value, name, context=context)
[9936]551
552class ComboCardDataExporter(grok.GlobalUtility, StudentExporterBase):
[12862]553    """Like all other exporters the Combo Card Data Exporter first filters the
554    set of students by searching the students catalog. Then it exports some
555    student base data which are neccessary to print for the Interswitch combo
556    card (identity card for students). The output contains a ``passport_path``
557    column which contains the filesystem path of the passport image file.
558    If no path is given, no passport image file exists.
[9936]559    """
560    grok.name('combocard')
561
562    fields = ('display_fullname',
[9937]563              'student_id','matric_number',
564              'certificate', 'faculty', 'department', 'passport_path')
[9936]565    title = _(u'Combo Card Data')
566
567    def mangle_value(self, value, name, context=None):
[12862]568        """The mangler determines the titles of faculty, department
569        and certificate. It also computes the path of passport image file
570        stored in the filesystem.
571        """
[9936]572        certificate = context['studycourse'].certificate
573        if name == 'certificate' and certificate is not None:
574            value = certificate.title
575        if name == 'department' and certificate is not None:
[10650]576            value = certificate.__parent__.__parent__.longtitle
[9936]577        if name == 'faculty' and certificate is not None:
[10650]578            value = certificate.__parent__.__parent__.__parent__.longtitle
[9937]579        if name == 'passport_path' and certificate is not None:
[12862]580            file_id = IFileStoreNameChooser(context).chooseName(
581                attr='passport.jpg')
[9937]582            os_path = getUtility(IExtFileStore)._pathFromFileID(file_id)
583            if not os.path.exists(os_path):
584                value = None
585            else:
586                value = '/'.join(os_path.split('/')[-4:])
[9936]587        return super(
588            ComboCardDataExporter, self).mangle_value(
589            value, name, context=context)
Note: See TracBrowser for help on using the repository browser.