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

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

Fix mangle_value method of DataForLecturerExporter?.

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