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

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

Make some adjustments to payment exporters and importers: Do not allow to import display_item, use IPayer adapter, fix docstring and more.

  • Property svn:keywords set to Id
File size: 24.2 KB
RevLine 
[8057]1## $Id: export.py 13871 2016-05-31 17:57:03Z 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
[13766]351class DataForLecturerExporter(CourseTicketExporter):
352    """
353    """
354    grok.name('lecturer')
[8342]355
[13766]356    fields = ('matric_number', 'reg_number', 'student_id', 'display_fullname',
357              'level', 'code', 'level_session', 'score')
358
359    title = _(u'Data for Lecturer')
360
361    def mangle_value(self, value, name, context=None):
362        """The mangler determines the student's id and fullname.
363        """
364        if context is not None:
365            student = context.student
366            if name in ('student_id', 'fullname') and student is not None:
367                value = getattr(student, name, None)
368        return super(
369            CourseTicketExporter, self).mangle_value(
370            value, name, context=context)
371
[12865]372class StudentPaymentExporter(grok.GlobalUtility, StudentExporterBase):
373    """The Student Payment Exporter first filters the set of students
[12862]374    by searching the students catalog. Then it exports student payment
375    tickets by iterating over the items of the student's ``payments``
376    container. If the payment period is given only tickets, which were
377    paid in payment period, are considered for export.
[8371]378    """
379    grok.name('studentpayments')
380
381    fields = tuple(
[8493]382        sorted(iface_names(
[9984]383            IStudentOnlinePayment, exclude_attribs=False,
[13871]384            omit=['display_item', 'certificate', 'student']))) + (
[13641]385            'student_id','state','current_session')
[8576]386    title = _(u'Student Payments')
[8371]387
[9844]388    def filter_func(self, x, **kw):
[11730]389        return get_payments(x, **kw)
[9802]390
[8371]391    def mangle_value(self, value, name, context=None):
[12862]392        """The mangler determines the student's id, registration
393        state and current session.
[8371]394        """
395        if context is not None:
[8736]396            student = context.student
[10232]397            if name in ['student_id','state',
398                        'current_session'] and student is not None:
[8371]399                value = getattr(student, name, None)
400        return super(
[12865]401            StudentPaymentExporter, self).mangle_value(
[8371]402            value, name, context=context)
403
[12971]404class StudentUnpaidPaymentExporter(StudentPaymentExporter):
405    """The Student Unpaid Payment Exporter works just like the
406    Student Payments Exporter but it exports only unpaid tickets.
407    This exporter is designed for finding and finally purging outdated
408    payment ticket.
409    """
410    grok.name('studentunpaidpayments')
411
412    title = _(u'Student Unpaid Payments')
413
414    def filter_func(self, x, **kw):
415        return get_payments(x, p_state='unpaid', **kw)
416
[12865]417class DataForBursaryExporter(StudentPaymentExporter):
[12862]418    """The DataForBursary Exporter works just like the Student Payments Exporter
419    but it exports much more information about the student. It combines
420    payment and student data in one table in order to spare postprocessing of
421    two seperate export files. The exporter is primarily used by bursary
422    officers who have exclusively access to this exporter.
[10233]423    """
424    grok.name('bursary')
425
[10296]426    def filter_func(self, x, **kw):
[12971]427        return get_payments(x, p_state='paid', **kw)
[10296]428
[10233]429    fields = tuple(
430        sorted(iface_names(
431            IStudentOnlinePayment, exclude_attribs=False,
[13871]432            omit=['display_item', 'certificate', 'student']))) + (
[11702]433            'student_id','matric_number','reg_number',
[10236]434            'firstname', 'middlename', 'lastname',
435            'state','current_session',
436            'entry_session', 'entry_mode',
[11702]437            'faccode', 'depcode','certcode')
[10233]438    title = _(u'Payment Data for Bursary')
439
440    def mangle_value(self, value, name, context=None):
[12862]441        """The mangler fetches the student data.
[10233]442        """
443        if context is not None:
444            student = context.student
[10236]445            if name in [
[11702]446                'student_id','matric_number', 'reg_number',
[10236]447                'firstname', 'middlename', 'lastname',
[11702]448                'state', 'current_session',
[10236]449                'entry_session', 'entry_mode',
[11702]450                'faccode', 'depcode', 'certcode'] and student is not None:
[10233]451                value = getattr(student, name, None)
452        return super(
[12865]453            StudentPaymentExporter, self).mangle_value(
[10233]454            value, name, context=context)
455
[12865]456class BedTicketExporter(grok.GlobalUtility, StudentExporterBase):
457    """The Bed Ticket Exporter first filters the set of students
[12862]458    by searching the students catalog. Then it exports bed
459    tickets by iterating over the items of the student's ``accommodation``
460    container.
[9427]461    """
462    grok.name('bedtickets')
463
464    fields = tuple(
465        sorted(iface_names(
[9984]466            IBedTicket, exclude_attribs=False,
[13314]467            omit=['display_coordinates', 'maint_payment_made']))) + (
[9984]468            'student_id', 'actual_bed_type')
[9427]469    title = _(u'Bed Tickets')
470
[9844]471    def filter_func(self, x, **kw):
[9802]472        return get_bedtickets(x)
473
[9427]474    def mangle_value(self, value, name, context=None):
[12862]475        """The mangler determines the student id and the type of the bed
476        which has been booked in the ticket.
[9427]477        """
478        if context is not None:
479            student = context.student
480            if name in ['student_id'] and student is not None:
481                value = getattr(student, name, None)
482        if name == 'bed' and value is not None:
483            value = getattr(value, 'bed_id', None)
484        if name == 'actual_bed_type':
485            value = getattr(getattr(context, 'bed', None), 'bed_type')
486        return super(
[12865]487            BedTicketExporter, self).mangle_value(
[9427]488            value, name, context=context)
489
[12079]490class StudentPaymentsOverviewExporter(StudentExporter):
[12862]491    """The Student Payments Overview Exporter first filters the set of students
492    by searching the students catalog. Then it exports some student base data
493    together with the total school fee amount paid in each year over a
494    predefined year range (current year - 9, ... , current year + 1).
[9574]495    """
496    grok.name('paymentsoverview')
497
498    curr_year = datetime.now().year
[13639]499    year_range = range(curr_year - 10, curr_year + 1)
[9574]500    year_range_tuple = tuple([str(year) for year in year_range])
501
[9983]502    fields = ('student_id', 'matric_number', 'display_fullname',
[9574]503        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
[13641]504        'current_level', 'current_session', 'current_mode',
505        'entry_session', 'reg_number'
[9574]506        ) + year_range_tuple
507    title = _(u'Student Payments Overview')
508
509    def mangle_value(self, value, name, context=None):
[12862]510        """The mangler summarizes the school fee amounts made per year. It
511        iterates over all paid school fee payment tickets and
512        adds together the amounts paid in a year. Waived payments
513        are marked ``waived``.
514        """
[9574]515        if name in self.year_range_tuple and context is not None:
[11661]516            value = 0
[9574]517            for ticket in context['payments'].values():
[12568]518                if ticket.p_category == 'schoolfee' and \
[9574]519                    ticket.p_session == int(name):
[12568]520                    if ticket.p_state == 'waived':
521                        value = 'waived'
522                        break
523                    if ticket.p_state == 'paid':
524                        try:
525                            value += ticket.amount_auth
526                        except TypeError:
527                            pass
[11661]528            if value == 0:
529                value = ''
[9574]530        return super(
[12079]531            StudentExporter, self).mangle_value(
[9734]532            value, name, context=context)
[9744]533
[12079]534class StudentStudyLevelsOverviewExporter(StudentExporter):
[12862]535    """The Student Study Levels Overview Exporter first filters the set of
536    students by searching the students catalog. Then it exports some student
537    base data together with the session key of registered levels.
538    Sample output:
539
540    header: ``...100,110,120,200,210,220,300...``
541
542    data: ``...2010,,,2011,2012,,2013...``
543
544    This csv data string means that level 100 was registered in session
545    2010/2011, level 200 in session 2011/2012, level 210 (200 on 1st probation)
546    in session 2012/2013 and level 300 in session 2013/2014.
[9744]547    """
548    grok.name('studylevelsoverview')
549
[9787]550    avail_levels = tuple([str(x) for x in study_levels(None)])
[9744]551
552    fields = ('student_id', ) + (
553        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
[9761]554        'entry_session', 'current_level', 'current_session',
[9787]555        ) + avail_levels
[9744]556    title = _(u'Student Study Levels Overview')
557
558    def mangle_value(self, value, name, context=None):
[12862]559        """The mangler checks if a given level has been registered. It returns
560        the ``level_session`` attribute of the student study level object
561        if the named level exists.
562        """
[9787]563        if name in self.avail_levels and context is not None:
[9744]564            value = ''
565            for level in context['studycourse'].values():
566                if level.level == int(name):
[9761]567                    value = '%s' % level.level_session
[9744]568                    break
569        return super(
[12079]570            StudentExporter, self).mangle_value(
[9744]571            value, name, context=context)
[9936]572
573class ComboCardDataExporter(grok.GlobalUtility, StudentExporterBase):
[12862]574    """Like all other exporters the Combo Card Data Exporter first filters the
575    set of students by searching the students catalog. Then it exports some
576    student base data which are neccessary to print for the Interswitch combo
577    card (identity card for students). The output contains a ``passport_path``
578    column which contains the filesystem path of the passport image file.
579    If no path is given, no passport image file exists.
[9936]580    """
581    grok.name('combocard')
582
583    fields = ('display_fullname',
[9937]584              'student_id','matric_number',
585              'certificate', 'faculty', 'department', 'passport_path')
[9936]586    title = _(u'Combo Card Data')
587
588    def mangle_value(self, value, name, context=None):
[12862]589        """The mangler determines the titles of faculty, department
590        and certificate. It also computes the path of passport image file
591        stored in the filesystem.
592        """
[9936]593        certificate = context['studycourse'].certificate
594        if name == 'certificate' and certificate is not None:
595            value = certificate.title
596        if name == 'department' and certificate is not None:
[10650]597            value = certificate.__parent__.__parent__.longtitle
[9936]598        if name == 'faculty' and certificate is not None:
[10650]599            value = certificate.__parent__.__parent__.__parent__.longtitle
[9937]600        if name == 'passport_path' and certificate is not None:
[12862]601            file_id = IFileStoreNameChooser(context).chooseName(
602                attr='passport.jpg')
[9937]603            os_path = getUtility(IExtFileStore)._pathFromFileID(file_id)
604            if not os.path.exists(os_path):
605                value = None
606            else:
607                value = '/'.join(os_path.split('/')[-4:])
[9936]608        return super(
609            ComboCardDataExporter, self).mangle_value(
610            value, name, context=context)
Note: See TracBrowser for help on using the repository browser.