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

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

StudentPaymentsOverviewExporter? should export reg_number.

  • Property svn:keywords set to Id
File size: 23.4 KB
RevLine 
[8057]1## $Id: export.py 13641 2016-01-20 17:18:11Z 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']))) + (
[13641]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
[13639]479    year_range = range(curr_year - 10, curr_year + 1)
[9574]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',
[13641]484        'current_level', 'current_session', 'current_mode',
485        'entry_session', 'reg_number'
[9574]486        ) + year_range_tuple
487    title = _(u'Student Payments Overview')
488
489    def mangle_value(self, value, name, context=None):
[12862]490        """The mangler summarizes the school fee amounts made per year. It
491        iterates over all paid school fee payment tickets and
492        adds together the amounts paid in a year. Waived payments
493        are marked ``waived``.
494        """
[9574]495        if name in self.year_range_tuple and context is not None:
[11661]496            value = 0
[9574]497            for ticket in context['payments'].values():
[12568]498                if ticket.p_category == 'schoolfee' and \
[9574]499                    ticket.p_session == int(name):
[12568]500                    if ticket.p_state == 'waived':
501                        value = 'waived'
502                        break
503                    if ticket.p_state == 'paid':
504                        try:
505                            value += ticket.amount_auth
506                        except TypeError:
507                            pass
[11661]508            if value == 0:
509                value = ''
[9574]510        return super(
[12079]511            StudentExporter, self).mangle_value(
[9734]512            value, name, context=context)
[9744]513
[12079]514class StudentStudyLevelsOverviewExporter(StudentExporter):
[12862]515    """The Student Study Levels Overview Exporter first filters the set of
516    students by searching the students catalog. Then it exports some student
517    base data together with the session key of registered levels.
518    Sample output:
519
520    header: ``...100,110,120,200,210,220,300...``
521
522    data: ``...2010,,,2011,2012,,2013...``
523
524    This csv data string means that level 100 was registered in session
525    2010/2011, level 200 in session 2011/2012, level 210 (200 on 1st probation)
526    in session 2012/2013 and level 300 in session 2013/2014.
[9744]527    """
528    grok.name('studylevelsoverview')
529
[9787]530    avail_levels = tuple([str(x) for x in study_levels(None)])
[9744]531
532    fields = ('student_id', ) + (
533        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
[9761]534        'entry_session', 'current_level', 'current_session',
[9787]535        ) + avail_levels
[9744]536    title = _(u'Student Study Levels Overview')
537
538    def mangle_value(self, value, name, context=None):
[12862]539        """The mangler checks if a given level has been registered. It returns
540        the ``level_session`` attribute of the student study level object
541        if the named level exists.
542        """
[9787]543        if name in self.avail_levels and context is not None:
[9744]544            value = ''
545            for level in context['studycourse'].values():
546                if level.level == int(name):
[9761]547                    value = '%s' % level.level_session
[9744]548                    break
549        return super(
[12079]550            StudentExporter, self).mangle_value(
[9744]551            value, name, context=context)
[9936]552
553class ComboCardDataExporter(grok.GlobalUtility, StudentExporterBase):
[12862]554    """Like all other exporters the Combo Card Data Exporter first filters the
555    set of students by searching the students catalog. Then it exports some
556    student base data which are neccessary to print for the Interswitch combo
557    card (identity card for students). The output contains a ``passport_path``
558    column which contains the filesystem path of the passport image file.
559    If no path is given, no passport image file exists.
[9936]560    """
561    grok.name('combocard')
562
563    fields = ('display_fullname',
[9937]564              'student_id','matric_number',
565              'certificate', 'faculty', 'department', 'passport_path')
[9936]566    title = _(u'Combo Card Data')
567
568    def mangle_value(self, value, name, context=None):
[12862]569        """The mangler determines the titles of faculty, department
570        and certificate. It also computes the path of passport image file
571        stored in the filesystem.
572        """
[9936]573        certificate = context['studycourse'].certificate
574        if name == 'certificate' and certificate is not None:
575            value = certificate.title
576        if name == 'department' and certificate is not None:
[10650]577            value = certificate.__parent__.__parent__.longtitle
[9936]578        if name == 'faculty' and certificate is not None:
[10650]579            value = certificate.__parent__.__parent__.__parent__.longtitle
[9937]580        if name == 'passport_path' and certificate is not None:
[12862]581            file_id = IFileStoreNameChooser(context).chooseName(
582                attr='passport.jpg')
[9937]583            os_path = getUtility(IExtFileStore)._pathFromFileID(file_id)
584            if not os.path.exists(os_path):
585                value = None
586            else:
587                value = '/'.join(os_path.split('/')[-4:])
[9936]588        return super(
589            ComboCardDataExporter, self).mangle_value(
590            value, name, context=context)
Note: See TracBrowser for help on using the repository browser.