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

Last change on this file since 16827 was 16827, checked in by Henrik Bettermann, 3 years ago

Add exporters for previous study course data.

  • Property svn:keywords set to Id
File size: 40.7 KB
RevLine 
[8057]1## $Id: export.py 16827 2022-02-22 12:48:36Z 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
[16679]22from copy import deepcopy
[13156]23from datetime import datetime, timedelta
[9937]24from zope.component import getUtility
[11757]25from waeup.kofa.interfaces import (
26    IExtFileStore, IFileStoreNameChooser, IKofaUtils)
[7944]27from waeup.kofa.interfaces import MessageFactory as _
[15873]28from waeup.kofa.university.interfaces import ICertificateCourse, ICourse
[9843]29from waeup.kofa.students.catalog import StudentsQuery, CourseTicketsQuery
[8015]30from waeup.kofa.students.interfaces import (
[8371]31    IStudent, IStudentStudyCourse, IStudentStudyLevel, ICourseTicket,
[9427]32    IStudentOnlinePayment, ICSVStudentExporter, IBedTicket)
[9787]33from waeup.kofa.students.vocabularies import study_levels
[7944]34from waeup.kofa.utils.batching import ExporterBase
[11757]35from waeup.kofa.utils.helpers import iface_names, to_timezone
[7944]36
[8400]37
[9787]38def get_students(site, stud_filter=StudentsQuery()):
[8414]39    """Get all students registered in catalog in `site`.
[7944]40    """
[9787]41    return stud_filter.query()
[8414]42
[16827]43def get_studycourses(students, previous=0):
[8414]44    """Get studycourses of `students`.
45    """
[16827]46    studycourse = 'studycourse'
47    if previous == 1:
48        studycourse = 'studycourse_1'
49    elif previous == 2:
50        studycourse = 'studycourse_2'
51    for x in students:
52        if x.get(studycourse, None) is None:
53            continue
54        yield x.get(studycourse, None)
[8414]55
[16827]56def get_levels(students, previous=0, **kw):
[8414]57    """Get all studylevels of `students`.
58    """
59    levels = []
[15918]60    level_session = kw.get('level_session', None)
[16827]61    for course in get_studycourses(students, previous):
[8414]62        for level in course.values():
[15918]63            if level_session not in ('all', None) and \
64                int(level_session) != level.level_session:
65                continue
[8414]66            levels.append(level)
67    return levels
68
[16827]69def get_tickets(students, previous=0, **kw):
[9844]70    """Get course tickets of `students`.
[10017]71    If code is passed through, filter course tickets
[14984]72    which belong to this course code and meet level=level
73    and level_session=level_session.
[15546]74    If not, but ct_level, ct_session and ct_semester
[14984]75    are passed through, filter course tickets
[15546]76    which meet level==ct_level, level_session==ct_session
77    and semester==ct_semester.
[8414]78    """
79    tickets = []
[9844]80    code = kw.get('code', None)
[10017]81    level = kw.get('level', None)
[14984]82    session = kw.get('session', None)
83    ct_level = kw.get('ct_level', None)
84    ct_session = kw.get('ct_session', None)
[15546]85    ct_semester = kw.get('ct_semester', None)
[9844]86    if code is None:
[16827]87        for level_obj in get_levels(students, previous, **kw):
[10017]88            for ticket in level_obj.values():
[14984]89                if ct_level not in ('all', None):
[16812]90                    if level_obj.level in (10, 999, 1000, None)  \
[14984]91                        and int(ct_level) != level_obj.level:
92                        continue
93                    if level_obj.level not in range(
94                        int(ct_level), int(ct_level)+100, 10):
95                        continue
96                if ct_session not in ('all', None) and \
97                    int(ct_session) != level_obj.level_session:
98                    continue
[15546]99                if ct_semester not in ('all', None) and \
100                    int(ct_semester) != ticket.semester:
101                    continue
[9844]102                tickets.append(ticket)
103    else:
[16827]104        for level_obj in get_levels(students, previous, **kw):
[10017]105            for ticket in level_obj.values():
106                if ticket.code != code:
107                    continue
[14984]108                if level not in ('all', None):
[16812]109                    if level_obj.level in (10, 999, 1000, None)  \
[11483]110                        and int(level) != level_obj.level:
111                        continue
[14984]112                    if level_obj.level not in range(
113                        int(level), int(level)+100, 10):
[11483]114                        continue
[14984]115                if session not in ('all', None) and \
116                    int(session) != level_obj.level_session:
[10017]117                    continue
118                tickets.append(ticket)
[8414]119    return tickets
120
[16827]121def get_tickets_for_lecturer(students, previous=0, **kw):
[13894]122    """Get course tickets of `students`.
123    Filter course tickets which belong to this course code and
[16412]124    which are editable by lecturers. The latter requirement was disabled
125    on 10/03/21.
[13894]126    """
127    tickets = []
128    code = kw.get('code', None)
129    level_session = kw.get('session', None)
130    level = kw.get('level', None)
[16827]131    for level_obj in get_levels(students, previous, **kw):
[13894]132        for ticket in level_obj.values():
133            if ticket.code != code:
134                continue
[16412]135            # Disabled on 10/03/21
136            #if not ticket.editable_by_lecturer:
137            #    continue
[14984]138            if level not in ('all', None):
[16812]139                if level_obj.level in (10, 999, 1000, None)  \
[13894]140                    and int(level) != level_obj.level:
141                    continue
[14984]142                if level_obj.level not in range(int(level), int(level)+100, 10):
[13894]143                    continue
[14984]144            if level_session not in ('all', None) and \
[13894]145                int(level_session) != level_obj.level_session:
146                continue
147            tickets.append(ticket)
148    return tickets
149
[15873]150def get_outstanding(students, **kw):
151    """Get students with outstanding certificate courses.
152    """
153    students_wo = []
154    for student in students:
155        certificate = getattr(
156            student.get('studycourse', None), 'certificate', None)
157        if certificate:
[16662]158            allticketcodes = list()
159            failedticketcodes = list() # taken but failed
160            nottakenticketcodes = list() # registered but not taken
161            missedticketcodes = list() # not registered
162            # collect failed, not taken and all courses
[15873]163            for level in student['studycourse'].values():
[16666]164                # check if already failed or not taken courses have been passed later
[16679]165                failedticketcodes_copy = deepcopy(failedticketcodes)
166                nottakenticketcodes_copy = deepcopy(nottakenticketcodes)
167                for code in failedticketcodes_copy:
168                    if code.strip('m_').strip('_m') in level.passed_params[6]:
[16666]169                        failedticketcodes.remove(code)
[16679]170                for code in nottakenticketcodes_copy:
[16666]171                    if code in level.passed_params[6]:
172                        nottakenticketcodes.remove(code)
[16662]173                failedticketcodes += level.passed_params[4].split()
174                nottakenticketcodes += level.passed_params[5].split()
[15873]175                for ticket in level.values():
176                    allticketcodes.append(ticket.code)
[16662]177            # collect missed tickets
[15873]178            for certcourse in certificate.values():
179                if certcourse.getCourseCode() not in allticketcodes:
[16662]180                    missedticketcodes.append(certcourse.__name__)
[15873]181            student_wo = (student, missedticketcodes,
182                          failedticketcodes, nottakenticketcodes)
183            students_wo.append(student_wo)
184    return students_wo
185
[14761]186def get_payments(students, p_states=None, **kw):
[15924]187    """Get all payment tickets of `students` within given payment_date period.
[10296]188    """
[11730]189    date_format = '%d/%m/%Y'
[10296]190    payments = []
[15043]191    p_start = kw.get('payments_start')
192    p_end = kw.get('payments_end')
[15042]193    paycat = kw.get('paycat')
[15055]194    paysession = kw.get('paysession')
[15042]195    for student in students:
196        for payment in student.get('payments', {}).values():
[15043]197            if p_start and p_end:
[15042]198                if not payment.payment_date:
199                    continue
[15043]200                payments_start = datetime.strptime(p_start, date_format)
201                payments_end = datetime.strptime(p_end, date_format)
[15042]202                tz = getUtility(IKofaUtils).tzinfo
203                payments_start = tz.localize(payments_start)
204                payments_end = tz.localize(payments_end) + timedelta(days=1)
205                payment_date = to_timezone(payment.payment_date, tz)
206                if payment_date < payments_start or payment_date > payments_end:
207                    continue
208            if p_states and not payment.p_state in p_states:
209                continue
210            if paycat not in ('all', None) and payment.p_category != paycat:
211                continue
[15297]212            if paysession not in ('all', None) \
213                and payment.p_session != int(paysession):
[15055]214                continue
[15042]215            payments.append(payment)
[10296]216    return payments
217
[9427]218def get_bedtickets(students):
[15924]219    """Get all bed tickets of `students`.
[9427]220    """
221    tickets = []
222    for student in students:
223        for ticket in student.get('accommodation', {}).values():
224            tickets.append(ticket)
225    return tickets
[8414]226
227class StudentExporterBase(ExporterBase):
228    """Exporter for students or related objects.
229    This is a baseclass.
230    """
231    grok.baseclass()
[8411]232    grok.implements(ICSVStudentExporter)
233    grok.provides(ICSVStudentExporter)
[8414]234
[9844]235    def filter_func(self, x, **kw):
[9802]236        return x
[9797]237
[9801]238    def get_filtered(self, site, **kw):
[9843]239        """Get students from a catalog filtered by keywords.
240        students_catalog is the default catalog. The keys must be valid
[9933]241        catalog index names.
[9843]242        Returns a simple empty list, a list with `Student`
243        objects or a catalog result set with `Student`
[9801]244        objects.
245
246        .. seealso:: `waeup.kofa.students.catalog.StudentsCatalog`
247
248        """
[9843]249        # Pass only given keywords to create FilteredCatalogQuery objects.
250        # This way we avoid
[9801]251        # trouble with `None` value ambivalences and queries are also
252        # faster (normally less indexes to ask). Drawback is, that
253        # developers must look into catalog to see what keywords are
254        # valid.
[9845]255        if kw.get('catalog', None) == 'coursetickets':
[9843]256            coursetickets = CourseTicketsQuery(**kw).query()
257            students = []
258            for ticket in coursetickets:
259                students.append(ticket.student)
260            return list(set(students))
[15042]261        # Payments can be filtered by payment date and payment category.
262        # These parameters are not keys of the catalog and must thus be
263        # removed from kw.
[11730]264        try:
265            del kw['payments_start']
266            del kw['payments_end']
[15042]267            del kw['paycat']
[15055]268            del kw['paysession']
[11730]269        except KeyError:
270            pass
[15055]271        # Coursetickets can be filtered by level and session.
[15042]272        # These parameters are not keys of the catalog and must thus be
273        # removed from kw.
[14984]274        try:
275            del kw['ct_level']
276            del kw['ct_session']
[15546]277            del kw['ct_semester']
[14984]278        except KeyError:
279            pass
[15918]280        # Studylevels can be filtered by level_session.
281        # This parameter is not a key of the catalog and must thus be
282        # removed from kw.
283        try:
284            del kw['level_session']
285        except KeyError:
286            pass
[9801]287        query = StudentsQuery(**kw)
[9797]288        return query.query()
289
[12518]290    def get_selected(self, site, selected):
291        """Get set of selected students.
292        Returns a simple empty list or a list with `Student`
293        objects.
294        """
295        students = []
296        students_container = site.get('students', {})
297        for id in selected:
298            student = students_container.get(id, None)
[14443]299            if student is None:
300                # try matric number
301                result = list(StudentsQuery(matric_number=id).query())
302                if result:
303                    student = result[0]
304                else:
305                    continue
306            students.append(student)
[12518]307        return students
308
[8414]309    def export(self, values, filepath=None):
310        """Export `values`, an iterable, as CSV file.
311        If `filepath` is ``None``, a raw string with CSV data is returned.
312        """
313        writer, outfile = self.get_csv_writer(filepath)
314        for value in values:
315            self.write_item(value, writer)
316        return self.close_outfile(filepath, outfile)
317
[9797]318    def export_all(self, site, filepath=None):
319        """Export students into filepath as CSV data.
320        If `filepath` is ``None``, a raw string with CSV data is returned.
[9763]321        """
[9802]322        return self.export(self.filter_func(get_students(site)), filepath)
[8414]323
[9797]324    def export_student(self, student, filepath=None):
325        return self.export(self.filter_func([student]), filepath=filepath)
[9734]326
[9802]327    def export_filtered(self, site, filepath=None, **kw):
328        """Export items denoted by `kw`.
329        If `filepath` is ``None``, a raw string with CSV data should
330        be returned.
331        """
332        data = self.get_filtered(site, **kw)
[9844]333        return self.export(self.filter_func(data, **kw), filepath=filepath)
[9802]334
[12518]335    def export_selected(self,site, filepath=None, **kw):
336        """Export data for selected set of students.
337        """
338        selected = kw.get('selected', [])
339        data = self.get_selected(site, selected)
340        return self.export(self.filter_func(data, **kw), filepath=filepath)
[9802]341
[12518]342
[12079]343class StudentExporter(grok.GlobalUtility, StudentExporterBase):
[12861]344    """The Student Exporter first filters the set of students by searching the
345    students catalog. Then it exports student base data of this set of students.
[8414]346    """
[7944]347    grok.name('students')
348
[9936]349    fields = tuple(sorted(iface_names(IStudent))) + (
[9253]350        'password', 'state', 'history', 'certcode', 'is_postgrad',
[16186]351        'current_level', 'current_session', 'entry_session')
[15920]352    title = _(u'Students (Data Backup)')
[7944]353
[8493]354    def mangle_value(self, value, name, context=None):
[12861]355        """The mangler prepares the history messages and adds a hash symbol at
356        the end of the phone number to avoid annoying automatic number
357        transformation by Excel or Calc."""
[8493]358        if name == 'history':
359            value = value.messages
[14957]360        if 'phone' in name and value is not None:
[8971]361            # Append hash '#' to phone numbers to circumvent
362            # unwanted excel automatic
[8947]363            value = str('%s#' % value)
[8493]364        return super(
[12079]365            StudentExporter, self).mangle_value(
[8493]366            value, name, context=context)
367
[15966]368class TrimmedDataExporter(grok.GlobalUtility, StudentExporterBase):
369    """The Student Trimmed Data Exporter first filters the set of students
370    by searching the students catalog. Then it exports a trimmed data set
371    of this set of students.
372    """
373    grok.name('trimmed')
[7944]374
[15966]375    fields = (
376        'student_id',
377        'matric_number',
378        'reg_number',
379        'firstname',
380        'middlename',
381        'lastname',
382        'sex',
383        'email',
384        'phone',
385        'nationality',
386        'date_of_birth',
387        'state',
388        'current_mode',
389        'certcode',
390        'faccode',
391        'depcode',
392        'current_level',
393        'current_session',
[16186]394        'current_verdict',
395        'entry_session')
[15966]396    title = _(u'Students (Trimmed Data)')
397
398    def mangle_value(self, value, name, context=None):
399        """The mangler prepares the history messages and adds a hash symbol at
400        the end of the phone number to avoid annoying automatic number
401        transformation by Excel or Calc."""
402        if 'phone' in name and value is not None:
403            # Append hash '#' to phone numbers to circumvent
404            # unwanted excel automatic
405            value = str('%s#' % value)
406        return super(
407            TrimmedDataExporter, self).mangle_value(
408            value, name, context=context)
409
[8414]410class StudentStudyCourseExporter(grok.GlobalUtility, StudentExporterBase):
[12861]411    """The Student Study Course Exporter first filters the set of students
412    by searching the students catalog. Then it exports the data of the current
413    study course container of each student from this set. It does
414    not export their content.
[7994]415    """
416    grok.name('studentstudycourses')
417
[8493]418    fields = tuple(sorted(iface_names(IStudentStudyCourse))) + ('student_id',)
[15920]419    title = _(u'Student Study Courses (Data Backup)')
[7994]420
[9844]421    def filter_func(self, x, **kw):
[9797]422        return get_studycourses(x)
423
[7994]424    def mangle_value(self, value, name, context=None):
[12861]425        """The mangler determines the certificate code and the student id.
[7994]426        """
427        if name == 'certificate' and value is not None:
428            # XXX: hopefully cert codes are unique site-wide
429            value = value.code
[8493]430        if name == 'student_id' and context is not None:
[8736]431            student = context.student
[8493]432            value = getattr(student, name, None)
[7994]433        return super(
434            StudentStudyCourseExporter, self).mangle_value(
435            value, name, context=context)
436
[16827]437class FirstStudentStudyCourseExporter(StudentStudyCourseExporter):
438    """The First Student Study Course Exporter exports the first
439    study course if student was transferred.
440    """
441    grok.name('studentstudycourses_1')
[7994]442
[16827]443    title = _(u'First Student Study Courses (Data Backup)')
444
445    def filter_func(self, x, **kw):
446        return get_studycourses(x, 1)
447
448class SecondStudentStudyCourseExporter(StudentStudyCourseExporter):
449    """The Second Student Study Course Exporter exports the second
450    study course if student was transferred twice.
451    """
452    grok.name('studentstudycourses_2')
453
454    title = _(u'Second Student Study Courses (Data Backup)')
455
456    def filter_func(self, x, **kw):
457        return get_studycourses(x, 2)
458
[8414]459class StudentStudyLevelExporter(grok.GlobalUtility, StudentExporterBase):
[12861]460    """The Student Study Level Exporter first filters the set of students
461    by searching the students catalog. Then it exports the data of the student's
[12862]462    study level container data but not their content (course tickets).
[12861]463    The exporter iterates over all objects in the students' ``studycourse``
464    containers.
[8015]465    """
466    grok.name('studentstudylevels')
467
[8493]468    fields = tuple(sorted(iface_names(
[12873]469        IStudentStudyLevel))) + (
[9253]470        'student_id', 'number_of_tickets','certcode')
[15920]471    title = _(u'Student Study Levels (Data Backup)')
[8015]472
[9844]473    def filter_func(self, x, **kw):
[15918]474        return get_levels(x, **kw)
[9802]475
[8015]476    def mangle_value(self, value, name, context=None):
[12861]477        """The mangler determines the student id, nothing else.
[8015]478        """
[8493]479        if name == 'student_id' and context is not None:
[8736]480            student = context.student
[8493]481            value = getattr(student, name, None)
[8015]482        return super(
483            StudentStudyLevelExporter, self).mangle_value(
484            value, name, context=context)
485
[16827]486class FirstStudentStudyLevelExporter(StudentStudyLevelExporter):
487    """The First Student Study Level Exporter exports study levels of the
488    first study course if the student was transferred.
489    """
490    grok.name('studentstudylevels_1')
491
492    title = _(u'First Course Student Study Levels (Data Backup)')
493
494    def filter_func(self, x, **kw):
495        return get_levels(x, 1, **kw)
496
497class SecondStudentStudyLevelExporter(StudentStudyLevelExporter):
498    """The Second Student Study Level Exporter exports study levels of the
499    second tudy course if the student was transferred twice.
500    """
501    grok.name('studentstudylevels_2')
502
503    title = _(u'Second Course Student Study Levels (Data Backup)')
504
505    def filter_func(self, x, **kw):
506        return get_levels(x, 2, **kw)
507
[8414]508class CourseTicketExporter(grok.GlobalUtility, StudentExporterBase):
[13144]509    """The Course Ticket Exporter exports course tickets. Usually,
[12861]510    the exporter first filters the set of students by searching the
511    students catalog. Then it collects and iterates over all ``studylevel``
512    containers of the filtered student set and finally
513    iterates over all items inside these containers.
514
515    If the course code is passed through, the exporter uses a different
516    catalog. It searches for students in the course tickets catalog and
517    exports those course tickets which belong to the given course code and
[13263]518    also meet level and session passed through at the same time.
[12862]519    This happens if the exporter is called at course level in the academic
[12861]520    section.
[8342]521    """
522    grok.name('coursetickets')
523
[8493]524    fields = tuple(sorted(iface_names(ICourseTicket) +
[11484]525        ['level', 'code', 'level_session'])) + ('student_id',
526        'certcode', 'display_fullname')
[15920]527    title = _(u'Course Tickets (Data Backup)')
[8342]528
[9844]529    def filter_func(self, x, **kw):
530        return get_tickets(x, **kw)
[9802]531
[8342]532    def mangle_value(self, value, name, context=None):
[12861]533        """The mangler determines the student's id and fullname.
[8342]534        """
535        if context is not None:
[8736]536            student = context.student
[11484]537            if name in ('student_id', 'display_fullname') and student is not None:
[8342]538                value = getattr(student, name, None)
539        return super(
540            CourseTicketExporter, self).mangle_value(
541            value, name, context=context)
542
[16827]543class FirstCourseTicketExporter(CourseTicketExporter):
544    """The First Course Ticket Exporter exports course tickets of
545    first study courses if the student was transferred.
546    """
547    grok.name('coursetickets_1')
548
549    title = _(u'First Course Course Tickets (Data Backup)')
550
551    def filter_func(self, x, **kw):
552        return get_tickets(x, 1, **kw)
553
554class SecondCourseTicketExporter(CourseTicketExporter):
555    """The Second Course Ticket Exporter exports course tickets of
556    second study courses if the student was transferred twice.
557    """
558    grok.name('coursetickets_2')
559
560    title = _(u'Second Course Course Tickets (Data Backup)')
561
562    def filter_func(self, x, **kw):
563        return get_tickets(x, 2, **kw)
564
[15924]565class StudentPaymentExporter(grok.GlobalUtility, StudentExporterBase):
566    """The Student Payment Exporter first filters the set of students
567    by searching the students catalog. Then it exports student payment
568    tickets by iterating over the items of the student's ``payments``
569    container. If the payment period is given, only tickets, which were
570    paid in payment period, are considered for export.
571    """
572    grok.name('studentpayments')
573
574    fields = tuple(
575        sorted(iface_names(
576            IStudentOnlinePayment, exclude_attribs=False,
[16431]577            omit=['display_item', 'certificate', 'student', 'p_option']))) + (
[15924]578            'student_id','state','current_session')
579    title = _(u'Student Payments (Data Backup)')
580
581    def filter_func(self, x, **kw):
582        return get_payments(x, **kw)
583
584    def mangle_value(self, value, name, context=None):
585        """The mangler determines the student's id, registration
586        state and current session.
587        """
588        if context is not None:
589            student = context.student
[16351]590            if name in ['student_id','state','entry_session',
[15924]591                        'current_session'] and student is not None:
592                value = getattr(student, name, None)
593        return super(
594            StudentPaymentExporter, self).mangle_value(
595            value, name, context=context)
596
[15979]597class StudentTrimmedPaymentExporter(grok.GlobalUtility, StudentExporterBase):
598    """The Student Trimmed Payment Exporter is a slightly customized version
599    of the StudentPaymentExporter requested by Uniben.
600    """
601    grok.name('trimmedpayments')
602
603    fields = tuple(
604        sorted(iface_names(
605            IStudentOnlinePayment, exclude_attribs=False,
[16431]606            omit=['display_item', 'certificate', 'student', 'ac', 'p_option']))) + (
[15979]607            'student_id','faccode', 'depcode', 'state','current_session')
608    title = _(u'Student Payments (Sorted Data)')
609
610    def filter_func(self, x, **kw):
611        payments = get_payments(x, **kw)
612        return sorted([payment for payment in payments],
[15996]613            key=lambda payment: str(payment.p_category) + str(payment.student.faccode)
[15979]614                + str(payment.student.depcode) + str(payment.p_item))
615
616    def mangle_value(self, value, name, context=None):
617        """The mangler determines the student's id, registration
618        state and current session.
619        """
620        if context is not None:
621            student = context.student
622            if name in ['student_id','state', 'faccode', 'depcode',
623                        'current_session'] and student is not None:
624                value = getattr(student, name, None)
625        return super(
626            StudentTrimmedPaymentExporter, self).mangle_value(
627            value, name, context=context)
628
[13894]629class DataForLecturerExporter(grok.GlobalUtility, StudentExporterBase):
[15049]630    """The Data for Lecturer Exporter searches for students in the course
631    tickets catalog and exports those course tickets which belong to the
632    given course code, meet level and session passed through at the
[16412]633    same time, and which are editable by lecturers (disabled on 10/03/21).
634    This exporter can only be called at course level in the academic section.
[13766]635    """
636    grok.name('lecturer')
[8342]637
[13894]638    fields = ('matric_number', 'student_id', 'display_fullname',
[13766]639              'level', 'code', 'level_session', 'score')
640
641    title = _(u'Data for Lecturer')
642
[13894]643    def filter_func(self, x, **kw):
[16006]644        tickets = get_tickets_for_lecturer(x, **kw)
645        return sorted([ticket for ticket in tickets],
646            key=lambda ticket: str(ticket.fcode) + str(ticket.dcode)
647                + str(ticket.student.matric_number))
[13894]648
[13766]649    def mangle_value(self, value, name, context=None):
650        """The mangler determines the student's id and fullname.
651        """
652        if context is not None:
653            student = context.student
[13885]654            if name in ('matric_number',
655                        'reg_number',
656                        'student_id',
657                        'display_fullname',) and student is not None:
[13766]658                value = getattr(student, name, None)
659        return super(
[13894]660            DataForLecturerExporter, self).mangle_value(
[13766]661            value, name, context=context)
662
[15924]663class OutstandingCoursesExporter(grok.GlobalUtility, StudentExporterBase):
[15873]664    """The Student Outstanding Courses Exporter first filters the set of
665    students by searching the students catalog. Then it exports students with
666    lists of outstanding courses, i.e. courses which the student has
667    missed (not registered at all), failed (registered but not passed)
668    or nottaken (registered but not taken).
669    """
[15920]670    grok.name('outstandingcourses')
[15873]671
[15963]672    fields = ('student_id', 'matric_number', 'certcode', 'display_fullname',
673              'missed', 'failed', 'nottaken')
[15920]674    title = _(u'Outstanding Courses')
[15873]675
676    def filter_func(self, x, **kw):
677        return get_outstanding(x, **kw)
678
679    def mangle_value(self, value, name, context=None):
680        """The mangler determines the student's id, fullname and certcode,
681        and it collects the lists of outstanding courses.
682        """
683        if context is not None:
[15963]684            if name in ('student_id', 'matric_number',
685                        'display_fullname', 'certcode'):
[15873]686                value = getattr(context[0], name, None)
687            elif name == 'missed':
[16662]688                value = ' '.join(context[1])
[15873]689            elif name == 'failed':
[16662]690                value = ' '.join(context[2])
[15873]691            elif name == 'nottaken':
[16662]692                value = ' '.join(context[3])
[15873]693        return super(
[15924]694            OutstandingCoursesExporter, self).mangle_value(
[15873]695            value, name, context=context)
696
[15924]697class UnpaidPaymentsExporter(StudentPaymentExporter):
698    """The Unpaid Payments Exporter works just like the
699    Student Payment (singular intended) Exporter  but it exports only
700    unpaid tickets. This exporter is designed for finding and finally
701    purging outdated payment tickets.
[8371]702    """
[15920]703    grok.name('unpaidpayments')
[12971]704
[15920]705    title = _(u'Unpaid Payment Tickets')
[12971]706
707    def filter_func(self, x, **kw):
[14761]708        return get_payments(x, p_states=('unpaid',) , **kw)
[12971]709
[12865]710class DataForBursaryExporter(StudentPaymentExporter):
[15278]711    """The Data for Bursary Exporter works just like the Student Payment
712    Exporter but it exports much more information about the student. It combines
[12862]713    payment and student data in one table in order to spare postprocessing of
714    two seperate export files. The exporter is primarily used by bursary
[14761]715    officers who have exclusively access to this exporter. The exporter
[15792]716    exports ``paid``, ``waived`` and ``scholarship`` payment tickets.
[10233]717    """
718    grok.name('bursary')
719
[10296]720    def filter_func(self, x, **kw):
[15792]721        return get_payments(x, p_states=('paid', 'waived', 'scholarship'), **kw)
[10296]722
[10233]723    fields = tuple(
724        sorted(iface_names(
725            IStudentOnlinePayment, exclude_attribs=False,
[16431]726            omit=['display_item', 'certificate', 'student', 'p_option']))) + (
[11702]727            'student_id','matric_number','reg_number',
[16329]728            'firstname', 'middlename', 'lastname','sex',
[10236]729            'state','current_session',
730            'entry_session', 'entry_mode',
[13943]731            'faccode', 'depcode','certcode')
[10233]732    title = _(u'Payment Data for Bursary')
733
734    def mangle_value(self, value, name, context=None):
[12862]735        """The mangler fetches the student data.
[10233]736        """
737        if context is not None:
738            student = context.student
[10236]739            if name in [
[11702]740                'student_id','matric_number', 'reg_number',
[16333]741                'firstname', 'middlename', 'lastname','sex',
[11702]742                'state', 'current_session',
[10236]743                'entry_session', 'entry_mode',
[11702]744                'faccode', 'depcode', 'certcode'] and student is not None:
[10233]745                value = getattr(student, name, None)
746        return super(
[12865]747            StudentPaymentExporter, self).mangle_value(
[10233]748            value, name, context=context)
749
[15277]750class AccommodationPaymentsExporter(DataForBursaryExporter):
[15278]751    """The Accommodation Payments Exporter works like the Data for Bursary
[15792]752    Exporter above. The exporter exports ``paid``, ``waived`` and ``scholarship``
753    payment tickets with category ``bed_allocation`` or ``hostel_maintenance``.
[15278]754    The exporter is primarily used by accommodation officers who have
755    exclusively access to this exporter.
[15277]756    """
757    grok.name('accommodationpayments')
758
759    def filter_func(self, x, **kw):
760        kw['paycat'] = 'bed_allocation'
[15792]761        payments = get_payments(x, p_states=(
762          'paid', 'waived', 'scholarship'), **kw)
[15277]763        kw['paycat'] = 'hostel_maintenance'
[15792]764        payments += get_payments(x, p_states=(
765          'paid', 'waived', 'scholarship'), **kw)
[15277]766        return payments
767
768    title = _(u'Accommodation Payments')
769
[12865]770class BedTicketExporter(grok.GlobalUtility, StudentExporterBase):
771    """The Bed Ticket Exporter first filters the set of students
[12862]772    by searching the students catalog. Then it exports bed
773    tickets by iterating over the items of the student's ``accommodation``
774    container.
[9427]775    """
776    grok.name('bedtickets')
777
778    fields = tuple(
779        sorted(iface_names(
[9984]780            IBedTicket, exclude_attribs=False,
[13314]781            omit=['display_coordinates', 'maint_payment_made']))) + (
[9984]782            'student_id', 'actual_bed_type')
[15920]783    title = _(u'Bed Tickets (Data Backup)')
[9427]784
[9844]785    def filter_func(self, x, **kw):
[9802]786        return get_bedtickets(x)
787
[9427]788    def mangle_value(self, value, name, context=None):
[12862]789        """The mangler determines the student id and the type of the bed
790        which has been booked in the ticket.
[9427]791        """
792        if context is not None:
793            student = context.student
794            if name in ['student_id'] and student is not None:
795                value = getattr(student, name, None)
796        if name == 'bed' and value is not None:
797            value = getattr(value, 'bed_id', None)
798        if name == 'actual_bed_type':
[14395]799            value = getattr(getattr(context, 'bed', None), 'bed_type', None)
[9427]800        return super(
[12865]801            BedTicketExporter, self).mangle_value(
[9427]802            value, name, context=context)
803
[15047]804class SchoolFeePaymentsOverviewExporter(StudentExporter):
805    """The School Fee Payments Overview Exporter first filters the set of students
[12862]806    by searching the students catalog. Then it exports some student base data
807    together with the total school fee amount paid in each year over a
808    predefined year range (current year - 9, ... , current year + 1).
[9574]809    """
[15047]810    grok.name('sfpaymentsoverview')
[9574]811
812    curr_year = datetime.now().year
[14596]813    year_range = range(curr_year - 11, curr_year + 1)
[9574]814    year_range_tuple = tuple([str(year) for year in year_range])
[9983]815    fields = ('student_id', 'matric_number', 'display_fullname',
[9574]816        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
[13641]817        'current_level', 'current_session', 'current_mode',
818        'entry_session', 'reg_number'
[9574]819        ) + year_range_tuple
[15920]820    title = _(u'School Fee Payments Overview')
[9574]821
822    def mangle_value(self, value, name, context=None):
[12862]823        """The mangler summarizes the school fee amounts made per year. It
824        iterates over all paid school fee payment tickets and
825        adds together the amounts paid in a year. Waived payments
[15792]826        are marked ``waived`` and scholarship payments marked `scholarship`.
[12862]827        """
[9574]828        if name in self.year_range_tuple and context is not None:
[11661]829            value = 0
[9574]830            for ticket in context['payments'].values():
[12568]831                if ticket.p_category == 'schoolfee' and \
[9574]832                    ticket.p_session == int(name):
[12568]833                    if ticket.p_state == 'waived':
834                        value = 'waived'
835                        break
[15792]836                    if ticket.p_state == 'scholarship':
837                        value = 'scholarship'
838                        break
[12568]839                    if ticket.p_state == 'paid':
840                        try:
841                            value += ticket.amount_auth
842                        except TypeError:
843                            pass
[11661]844            if value == 0:
845                value = ''
[14367]846            elif isinstance(value, float):
847                value = round(value, 2)
[9574]848        return super(
[12079]849            StudentExporter, self).mangle_value(
[9734]850            value, name, context=context)
[9744]851
[15051]852class SessionPaymentsOverviewExporter(StudentExporter):
853    """The Session Payments Overview Exporter first filters the set of students
854    by searching the students catalog. Then it exports some student base data
[15060]855    together with the total amount paid in predefined payment categories
856    over the previous three session (referring to current academic session).
857    Sample output:
858
859    header: ``...schoolfee13,schoolfee14,schoolfee15,gown13,gown14,gown15...``
860
861    data: ``...2000.0,,3000.0,,,1000.0,...``
862
[15062]863    This csv data string means that the student paid 2000.0 school fee in 2013
864    and 3000.0 in 2015. S/He furthermore paid 1000.0 for gown rental in 2015.
[15051]865    """
866    grok.name('sessionpaymentsoverview')
867
868    paycats = ('schoolfee', 'clearance', 'gown', 'transcript')
[15060]869    regular_fields = ('student_id', 'matric_number', 'display_fullname',
[15051]870        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
871        'current_level', 'current_session', 'current_mode',
872        'entry_session', 'reg_number'
[15060]873        )
[15051]874    title = _(u'Session Payments Overview')
875
[15060]876    @property
877    def paycatyears(self):
878        cas = grok.getSite()['configuration'].current_academic_session
879        paycatyears = []
880        if cas:
[15918]881            year_range = range(cas-2, cas+1)
[15060]882            year_range_tuple = tuple([str(year)[2:] for year in year_range])
883            paycatyears = [
884                cat+year for cat in self.paycats for year in year_range_tuple]
885        return paycatyears
886
887    @property
888    def fields(self):
889        return self.regular_fields + tuple(self.paycatyears)
890
[15051]891    def mangle_value(self, value, name, context=None):
892        """
893        """
[15060]894        amounts = dict()
895        for catyear in self.paycatyears:
896            amounts[catyear] = 0.0
897        if name[:-2] in self.paycats and context is not None:
[15051]898            for ticket in context['payments'].values():
[15060]899                if ticket.p_category == name[:-2]:
[15051]900                    if ticket.p_state in ('waived', 'paid'):
[15060]901                        if str(ticket.p_session)[2:] == name[-2:]:
902                            amounts[name] += ticket.amount_auth
903            if amounts[name] == 0.0:
904                value = ''
905            elif isinstance(amounts[name], float):
906                value = round(amounts[name], 2)
[15051]907        return super(
908            StudentExporter, self).mangle_value(
909            value, name, context=context)
910
[15921]911class StudyLevelsOverviewExporter(StudentExporter):
[12862]912    """The Student Study Levels Overview Exporter first filters the set of
913    students by searching the students catalog. Then it exports some student
914    base data together with the session key of registered levels.
915    Sample output:
916
917    header: ``...100,110,120,200,210,220,300...``
918
919    data: ``...2010,,,2011,2012,,2013...``
920
921    This csv data string means that level 100 was registered in session
922    2010/2011, level 200 in session 2011/2012, level 210 (200 on 1st probation)
923    in session 2012/2013 and level 300 in session 2013/2014.
[9744]924    """
925    grok.name('studylevelsoverview')
926
[9787]927    avail_levels = tuple([str(x) for x in study_levels(None)])
[9744]928
929    fields = ('student_id', ) + (
930        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
[9761]931        'entry_session', 'current_level', 'current_session',
[9787]932        ) + avail_levels
[15920]933    title = _(u'Study Levels Overview')
[9744]934
935    def mangle_value(self, value, name, context=None):
[12862]936        """The mangler checks if a given level has been registered. It returns
937        the ``level_session`` attribute of the student study level object
938        if the named level exists.
939        """
[9787]940        if name in self.avail_levels and context is not None:
[9744]941            value = ''
942            for level in context['studycourse'].values():
943                if level.level == int(name):
[9761]944                    value = '%s' % level.level_session
[9744]945                    break
946        return super(
[12079]947            StudentExporter, self).mangle_value(
[9744]948            value, name, context=context)
[9936]949
950class ComboCardDataExporter(grok.GlobalUtility, StudentExporterBase):
[12862]951    """Like all other exporters the Combo Card Data Exporter first filters the
952    set of students by searching the students catalog. Then it exports some
953    student base data which are neccessary to print for the Interswitch combo
954    card (identity card for students). The output contains a ``passport_path``
955    column which contains the filesystem path of the passport image file.
956    If no path is given, no passport image file exists.
[9936]957    """
958    grok.name('combocard')
959
960    fields = ('display_fullname',
[9937]961              'student_id','matric_number',
962              'certificate', 'faculty', 'department', 'passport_path')
[9936]963    title = _(u'Combo Card Data')
964
965    def mangle_value(self, value, name, context=None):
[12862]966        """The mangler determines the titles of faculty, department
967        and certificate. It also computes the path of passport image file
968        stored in the filesystem.
969        """
[9936]970        certificate = context['studycourse'].certificate
971        if name == 'certificate' and certificate is not None:
972            value = certificate.title
973        if name == 'department' and certificate is not None:
[10650]974            value = certificate.__parent__.__parent__.longtitle
[9936]975        if name == 'faculty' and certificate is not None:
[10650]976            value = certificate.__parent__.__parent__.__parent__.longtitle
[9937]977        if name == 'passport_path' and certificate is not None:
[12862]978            file_id = IFileStoreNameChooser(context).chooseName(
979                attr='passport.jpg')
[9937]980            os_path = getUtility(IExtFileStore)._pathFromFileID(file_id)
981            if not os.path.exists(os_path):
982                value = None
983            else:
984                value = '/'.join(os_path.split('/')[-4:])
[9936]985        return super(
986            ComboCardDataExporter, self).mangle_value(
[15920]987            value, name, context=context)
988
[15921]989class TranscriptDataExporter(StudentExporter):
990    """The Transcript Data Exporter first filters the set of
[15920]991    students by searching the students catalog. Then it exports student data
992    along with their transcript data.
993    """
994    grok.name('transcriptdata')
995
996    fields = ('student_id', ) + (
997        'state', 'certcode', 'faccode', 'depcode',
998        'entry_session', 'current_level', 'current_session',
999        'transcript_data')
1000    title = _(u'Transcript Data')
1001
1002    def mangle_value(self, value, name, context=None):
[15924]1003        """The mangler determines and formats the transcript data.
[15920]1004        """
1005        if name == 'transcript_data':
1006            value = {}
1007            td = context['studycourse'].getTranscriptData()[0]
1008            for level in td:
1009                tickets_1 = ','.join(i.code for i in level['tickets_1'])
1010                tickets_2 = ','.join(i.code for i in level['tickets_2'])
1011                tickets_3 = ','.join(i.code for i in level['tickets_3'])
1012                value = "Level %s; 1st: %s; 2nd: %s; 3rd: %s; sgpa: %s" % (
1013                    level['level_key'], tickets_1, tickets_2,
1014                    tickets_3, level['sgpa'],
1015                    )
1016        return super(
[15921]1017            TranscriptDataExporter, self).mangle_value(
[9936]1018            value, name, context=context)
Note: See TracBrowser for help on using the repository browser.