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

Last change on this file since 11681 was 11662, checked in by Henrik Bettermann, 11 years ago

Catch TypeError? exception if amount_auth is None.

  • Property svn:keywords set to Id
File size: 17.8 KB
RevLine 
[8057]1## $Id: export.py 11662 2014-05-21 07:31:23Z 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
[9574]22from datetime import datetime
[9937]23from zope.component import getUtility
24from waeup.kofa.interfaces import IExtFileStore, IFileStoreNameChooser
[7944]25from waeup.kofa.interfaces import MessageFactory as _
[9843]26from waeup.kofa.students.catalog import StudentsQuery, CourseTicketsQuery
[8015]27from waeup.kofa.students.interfaces import (
[8371]28    IStudent, IStudentStudyCourse, IStudentStudyLevel, ICourseTicket,
[9427]29    IStudentOnlinePayment, ICSVStudentExporter, IBedTicket)
[9787]30from waeup.kofa.students.vocabularies import study_levels
[7944]31from waeup.kofa.utils.batching import ExporterBase
32from waeup.kofa.utils.helpers import iface_names
33
[8400]34#: A tuple containing all exporter names referring to students or
35#: subobjects thereof.
[9834]36EXPORTER_NAMES = ('students', 'studentstudycourses',
37        'studentstudylevels', 'coursetickets',
38        'studentpayments', 'bedtickets', 'paymentsoverview',
[10233]39        'studylevelsoverview', 'combocard', 'bursary')
[8400]40
[9787]41def get_students(site, stud_filter=StudentsQuery()):
[8414]42    """Get all students registered in catalog in `site`.
[7944]43    """
[9787]44    return stud_filter.query()
[8414]45
46def get_studycourses(students):
47    """Get studycourses of `students`.
48    """
49    return [x.get('studycourse', None) for x in students
50            if x is not None]
51
52def get_levels(students):
53    """Get all studylevels of `students`.
54    """
55    levels = []
56    for course in get_studycourses(students):
57        for level in course.values():
58            levels.append(level)
59    return levels
60
[9844]61def get_tickets(students, **kw):
62    """Get course tickets of `students`.
63
[10017]64    If code is passed through, filter course tickets
65    which belong to this course code and meets level
66    and level_session.
[8414]67    """
68    tickets = []
[9844]69    code = kw.get('code', None)
[10017]70    level = kw.get('level', None)
71    level_session = kw.get('level_session', None)
[9844]72    if code is None:
[10017]73        for level_obj in get_levels(students):
74            for ticket in level_obj.values():
[9844]75                tickets.append(ticket)
76    else:
[10017]77        for level_obj in get_levels(students):
78            for ticket in level_obj.values():
79                if ticket.code != code:
80                    continue
[11483]81                if level is not None:
82                    level = int(level)
83                    if level_obj.level in (10, 999, None)  \
84                        and int(level) != level_obj.level:
85                        continue
86                    if level_obj.level not in range(level, level+100, 10):
87                        continue
[10017]88                if level_session is not None and \
89                    int(level_session) != level_obj.level_session:
90                    continue
91                tickets.append(ticket)
[8414]92    return tickets
93
94def get_payments(students):
95    """Get all payments of `students`.
96    """
97    payments = []
98    for student in students:
99        for payment in student.get('payments', {}).values():
100            payments.append(payment)
101    return payments
102
[10296]103def get_paid_payments(students):
104    """Get all paid payments of `students`.
105    """
106    payments = []
107    for student in students:
108        for payment in student.get('payments', {}).values():
109            if payment.p_state == 'paid':
110                payments.append(payment)
111    return payments
112
[9427]113def get_bedtickets(students):
114    """Get all bedtickets of `students`.
115    """
116    tickets = []
117    for student in students:
118        for ticket in student.get('accommodation', {}).values():
119            tickets.append(ticket)
120    return tickets
[8414]121
122class StudentExporterBase(ExporterBase):
123    """Exporter for students or related objects.
124
125    This is a baseclass.
126    """
127    grok.baseclass()
[8411]128    grok.implements(ICSVStudentExporter)
129    grok.provides(ICSVStudentExporter)
[8414]130
[9844]131    def filter_func(self, x, **kw):
[9802]132        return x
[9797]133
[9801]134    def get_filtered(self, site, **kw):
[9843]135        """Get students from a catalog filtered by keywords.
[9801]136
[9843]137        students_catalog is the default catalog. The keys must be valid
[9933]138        catalog index names.
[9843]139        Returns a simple empty list, a list with `Student`
140        objects or a catalog result set with `Student`
[9801]141        objects.
142
143        .. seealso:: `waeup.kofa.students.catalog.StudentsCatalog`
144
145        """
[9843]146        # Pass only given keywords to create FilteredCatalogQuery objects.
147        # This way we avoid
[9801]148        # trouble with `None` value ambivalences and queries are also
149        # faster (normally less indexes to ask). Drawback is, that
150        # developers must look into catalog to see what keywords are
151        # valid.
[9845]152        if kw.get('catalog', None) == 'coursetickets':
[9843]153            coursetickets = CourseTicketsQuery(**kw).query()
154            students = []
155            for ticket in coursetickets:
156                students.append(ticket.student)
157            return list(set(students))
[9801]158        query = StudentsQuery(**kw)
[9797]159        return query.query()
160
[8414]161    def export(self, values, filepath=None):
162        """Export `values`, an iterable, as CSV file.
163
164        If `filepath` is ``None``, a raw string with CSV data is returned.
165        """
166        writer, outfile = self.get_csv_writer(filepath)
167        for value in values:
168            self.write_item(value, writer)
169        return self.close_outfile(filepath, outfile)
170
[9797]171    def export_all(self, site, filepath=None):
172        """Export students into filepath as CSV data.
173
174        If `filepath` is ``None``, a raw string with CSV data is returned.
[9763]175        """
[9802]176        return self.export(self.filter_func(get_students(site)), filepath)
[8414]177
[9797]178    def export_student(self, student, filepath=None):
179        return self.export(self.filter_func([student]), filepath=filepath)
[9734]180
[9802]181    def export_filtered(self, site, filepath=None, **kw):
182        """Export items denoted by `kw`.
[9797]183
[9802]184        If `filepath` is ``None``, a raw string with CSV data should
185        be returned.
186        """
187        data = self.get_filtered(site, **kw)
[9844]188        return self.export(self.filter_func(data, **kw), filepath=filepath)
[9802]189
190
[8414]191class StudentsExporter(grok.GlobalUtility, StudentExporterBase):
192    """Exporter for Students.
193    """
[7944]194    grok.name('students')
195
196    #: Fieldnames considered by this exporter
[9936]197    fields = tuple(sorted(iface_names(IStudent))) + (
[9253]198        'password', 'state', 'history', 'certcode', 'is_postgrad',
199        'current_level', 'current_session')
[7944]200
201    #: The title under which this exporter will be displayed
202    title = _(u'Students')
203
[8493]204    def mangle_value(self, value, name, context=None):
205        if name == 'history':
206            value = value.messages
[8971]207        if name == 'phone' and value is not None:
208            # Append hash '#' to phone numbers to circumvent
209            # unwanted excel automatic
[8947]210            value = str('%s#' % value)
[8493]211        return super(
212            StudentsExporter, self).mangle_value(
213            value, name, context=context)
214
[7944]215
[8414]216class StudentStudyCourseExporter(grok.GlobalUtility, StudentExporterBase):
[7994]217    """Exporter for StudentStudyCourses.
218    """
219    grok.name('studentstudycourses')
220
221    #: Fieldnames considered by this exporter
[8493]222    fields = tuple(sorted(iface_names(IStudentStudyCourse))) + ('student_id',)
[7994]223
224    #: The title under which this exporter will be displayed
225    title = _(u'Student Study Courses')
226
[9844]227    def filter_func(self, x, **kw):
[9797]228        return get_studycourses(x)
229
[7994]230    def mangle_value(self, value, name, context=None):
[8493]231        """Treat location values special.
[7994]232        """
233        if name == 'certificate' and value is not None:
234            # XXX: hopefully cert codes are unique site-wide
235            value = value.code
[8493]236        if name == 'student_id' and context is not None:
[8736]237            student = context.student
[8493]238            value = getattr(student, name, None)
[7994]239        return super(
240            StudentStudyCourseExporter, self).mangle_value(
241            value, name, context=context)
242
243
[8414]244class StudentStudyLevelExporter(grok.GlobalUtility, StudentExporterBase):
[8015]245    """Exporter for StudentStudyLevels.
246    """
247    grok.name('studentstudylevels')
248
249    #: Fieldnames considered by this exporter
[8493]250    fields = tuple(sorted(iface_names(
[9253]251        IStudentStudyLevel) + ['level'])) + (
252        'student_id', 'number_of_tickets','certcode')
[8015]253
254    #: The title under which this exporter will be displayed
255    title = _(u'Student Study Levels')
256
[9844]257    def filter_func(self, x, **kw):
[9802]258        return get_levels(x)
259
[8015]260    def mangle_value(self, value, name, context=None):
[8493]261        """Treat location values special.
[8015]262        """
[8493]263        if name == 'student_id' and context is not None:
[8736]264            student = context.student
[8493]265            value = getattr(student, name, None)
[8015]266        return super(
267            StudentStudyLevelExporter, self).mangle_value(
268            value, name, context=context)
269
[8414]270class CourseTicketExporter(grok.GlobalUtility, StudentExporterBase):
[8342]271    """Exporter for CourseTickets.
272    """
273    grok.name('coursetickets')
274
275    #: Fieldnames considered by this exporter
[8493]276    fields = tuple(sorted(iface_names(ICourseTicket) +
[11484]277        ['level', 'code', 'level_session'])) + ('student_id',
278        'certcode', 'display_fullname')
[8342]279
280    #: The title under which this exporter will be displayed
281    title = _(u'Course Tickets')
282
[9844]283    def filter_func(self, x, **kw):
284        return get_tickets(x, **kw)
[9802]285
[8342]286    def mangle_value(self, value, name, context=None):
287        """Treat location values special.
288        """
289        if context is not None:
[8736]290            student = context.student
[11484]291            if name in ('student_id', 'display_fullname') and student is not None:
[8342]292                value = getattr(student, name, None)
[11484]293            #if name == 'level':
294            #    value = getattr(context, 'level', lambda: None)
295            #if name == 'level_session':
296            #    value = getattr(context, 'level_session', lambda: None)
[8342]297        return super(
298            CourseTicketExporter, self).mangle_value(
299            value, name, context=context)
300
301
[9832]302class StudentPaymentsExporter(grok.GlobalUtility, StudentExporterBase):
[8371]303    """Exporter for OnlinePayment instances.
304    """
305    grok.name('studentpayments')
306
307    #: Fieldnames considered by this exporter
308    fields = tuple(
[8493]309        sorted(iface_names(
[9984]310            IStudentOnlinePayment, exclude_attribs=False,
311            omit=['display_item']))) + (
[10232]312            'student_id','state','current_session')
[8371]313
314    #: The title under which this exporter will be displayed
[8576]315    title = _(u'Student Payments')
[8371]316
[9844]317    def filter_func(self, x, **kw):
[9802]318        return get_payments(x)
319
[8371]320    def mangle_value(self, value, name, context=None):
321        """Treat location values special.
322        """
323        if context is not None:
[8736]324            student = context.student
[10232]325            if name in ['student_id','state',
326                        'current_session'] and student is not None:
[8371]327                value = getattr(student, name, None)
328        return super(
[9832]329            StudentPaymentsExporter, self).mangle_value(
[8371]330            value, name, context=context)
331
[10233]332class DataForBursaryExporter(StudentPaymentsExporter):
333    """Exporter for OnlinePayment instances.
334    """
335    grok.name('bursary')
336
[10296]337    def filter_func(self, x, **kw):
338        return get_paid_payments(x)
339
[10233]340    #: Fieldnames considered by this exporter
341    fields = tuple(
342        sorted(iface_names(
343            IStudentOnlinePayment, exclude_attribs=False,
344            omit=['display_item']))) + (
[10236]345            'student_id','matric_number',
346            'firstname', 'middlename', 'lastname',
347            'state','current_session',
348            'entry_session', 'entry_mode',
349            'faccode', 'depcode')
[10233]350
351    #: The title under which this exporter will be displayed
352    title = _(u'Payment Data for Bursary')
353
354    def mangle_value(self, value, name, context=None):
355        """Treat location values special.
356        """
357        if context is not None:
358            student = context.student
[10236]359            if name in [
360                'student_id','matric_number',
361                'firstname', 'middlename', 'lastname',
362                'state','current_session',
363                'entry_session', 'entry_mode',
364                'faccode', 'depcode'] and student is not None:
[10233]365                value = getattr(student, name, None)
366        return super(
367            StudentPaymentsExporter, self).mangle_value(
368            value, name, context=context)
369
[9427]370class BedTicketsExporter(grok.GlobalUtility, StudentExporterBase):
371    """Exporter for BedTicket instances.
372    """
373    grok.name('bedtickets')
374
375    #: Fieldnames considered by this exporter
376    fields = tuple(
377        sorted(iface_names(
[9984]378            IBedTicket, exclude_attribs=False,
379            omit=['display_coordinates']))) + (
380            'student_id', 'actual_bed_type')
[9427]381
382    #: The title under which this exporter will be displayed
383    title = _(u'Bed Tickets')
384
[9844]385    def filter_func(self, x, **kw):
[9802]386        return get_bedtickets(x)
387
[9427]388    def mangle_value(self, value, name, context=None):
389        """Treat location values and others special.
390        """
391        if context is not None:
392            student = context.student
393            if name in ['student_id'] and student is not None:
394                value = getattr(student, name, None)
395        if name == 'bed' and value is not None:
396            value = getattr(value, 'bed_id', None)
397        if name == 'actual_bed_type':
398            value = getattr(getattr(context, 'bed', None), 'bed_type')
399        return super(
400            BedTicketsExporter, self).mangle_value(
401            value, name, context=context)
402
[9574]403class StudentPaymentsOverviewExporter(StudentsExporter):
404    """Exporter for students with payment overview.
405    """
406    grok.name('paymentsoverview')
407
408    curr_year = datetime.now().year
409    year_range = range(curr_year - 9, curr_year + 1)
410    year_range_tuple = tuple([str(year) for year in year_range])
411
412    #: Fieldnames considered by this exporter
[9983]413    fields = ('student_id', 'matric_number', 'display_fullname',
[9574]414        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
[9807]415        'current_level', 'current_session', 'current_mode',
[9574]416        ) + year_range_tuple
417
418    #: The title under which this exporter will be displayed
419    title = _(u'Student Payments Overview')
420
421    def mangle_value(self, value, name, context=None):
422        if name in self.year_range_tuple and context is not None:
[11661]423            value = 0
[9574]424            for ticket in context['payments'].values():
425                if ticket.p_state == 'paid' and \
426                    ticket.p_category == 'schoolfee' and \
427                    ticket.p_session == int(name):
[11662]428                    try:
429                        value += ticket.amount_auth
430                    except TypeError:
431                        pass
[11661]432            if value == 0:
433                value = ''
[9574]434        return super(
435            StudentsExporter, self).mangle_value(
[9734]436            value, name, context=context)
[9744]437
438class StudentStudyLevelsOverviewExporter(StudentsExporter):
439    """Exporter for students with study level overview.
440    """
441    grok.name('studylevelsoverview')
442
[9787]443    avail_levels = tuple([str(x) for x in study_levels(None)])
[9744]444
445    #: Fieldnames considered by this exporter
446    fields = ('student_id', ) + (
447        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
[9761]448        'entry_session', 'current_level', 'current_session',
[9787]449        ) + avail_levels
[9744]450
451    #: The title under which this exporter will be displayed
452    title = _(u'Student Study Levels Overview')
453
454    def mangle_value(self, value, name, context=None):
[9787]455        if name in self.avail_levels and context is not None:
[9744]456            value = ''
457            for level in context['studycourse'].values():
458                if level.level == int(name):
[9761]459                    #value = '%s|%s|%s|%s' % (
460                    #    level.level_session,
461                    #    len(level),
462                    #    level.validated_by,
463                    #    level.level_verdict)
464                    value = '%s' % level.level_session
[9744]465                    break
466        return super(
467            StudentsExporter, self).mangle_value(
468            value, name, context=context)
[9936]469
470class ComboCardDataExporter(grok.GlobalUtility, StudentExporterBase):
471    """Exporter for Interswitch Combo Card Data.
472    """
473    grok.name('combocard')
474
475    #: Fieldnames considered by this exporter
476    fields = ('display_fullname',
[9937]477              'student_id','matric_number',
478              'certificate', 'faculty', 'department', 'passport_path')
[9936]479
480    #: The title under which this exporter will be displayed
481    title = _(u'Combo Card Data')
482
483    def mangle_value(self, value, name, context=None):
484        certificate = context['studycourse'].certificate
485        if name == 'certificate' and certificate is not None:
486            value = certificate.title
487        if name == 'department' and certificate is not None:
[10650]488            value = certificate.__parent__.__parent__.longtitle
[9936]489        if name == 'faculty' and certificate is not None:
[10650]490            value = certificate.__parent__.__parent__.__parent__.longtitle
[9937]491        if name == 'passport_path' and certificate is not None:
492            file_id = IFileStoreNameChooser(context).chooseName(attr='passport.jpg')
493            os_path = getUtility(IExtFileStore)._pathFromFileID(file_id)
494            if not os.path.exists(os_path):
495                value = None
496            else:
497                value = '/'.join(os_path.split('/')[-4:])
[9936]498        return super(
499            ComboCardDataExporter, self).mangle_value(
500            value, name, context=context)
Note: See TracBrowser for help on using the repository browser.