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

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

More documentation.

  • Property svn:keywords set to Id
File size: 21.1 KB
Line 
1## $Id: export.py 12861 2015-04-17 14:43:42Z 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##
18"""Exporters for student related stuff.
19"""
20import os
21import grok
22from datetime import datetime
23from zope.component import getUtility
24from waeup.kofa.interfaces import (
25    IExtFileStore, IFileStoreNameChooser, IKofaUtils)
26from waeup.kofa.interfaces import MessageFactory as _
27from waeup.kofa.students.catalog import StudentsQuery, CourseTicketsQuery
28from waeup.kofa.students.interfaces import (
29    IStudent, IStudentStudyCourse, IStudentStudyLevel, ICourseTicket,
30    IStudentOnlinePayment, ICSVStudentExporter, IBedTicket)
31from waeup.kofa.students.vocabularies import study_levels
32from waeup.kofa.utils.batching import ExporterBase
33from waeup.kofa.utils.helpers import iface_names, to_timezone
34
35
36def get_students(site, stud_filter=StudentsQuery()):
37    """Get all students registered in catalog in `site`.
38    """
39    return stud_filter.query()
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
56def get_tickets(students, **kw):
57    """Get course tickets of `students`.
58    If code is passed through, filter course tickets
59    which belong to this course code and meets level
60    and level_session.
61    """
62    tickets = []
63    code = kw.get('code', None)
64    level = kw.get('level', None)
65    level_session = kw.get('level_session', None)
66    if code is None:
67        for level_obj in get_levels(students):
68            for ticket in level_obj.values():
69                tickets.append(ticket)
70    else:
71        for level_obj in get_levels(students):
72            for ticket in level_obj.values():
73                if ticket.code != code:
74                    continue
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
82                if level_session is not None and \
83                    int(level_session) != level_obj.level_session:
84                    continue
85                tickets.append(ticket)
86    return tickets
87
88def get_payments(students, paid=False, **kw):
89    """Get all payments of `students` within given payment_date period.
90    """
91    date_format = '%d/%m/%Y'
92    payments = []
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)
99        tz = getUtility(IKofaUtils).tzinfo
100        payments_start = tz.localize(payments_start)
101        payments_end = tz.localize(payments_end)
102        if paid:
103            # Only paid tickets in payment period are considered
104            for student in students:
105                for payment in student.get('payments', {}).values():
106                    if payment.payment_date and payment.p_state == 'paid':
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)
111        else:
112            # All tickets in payment period are considered
113            for student in students:
114                for payment in student.get('payments', {}).values():
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)
120    else:
121        # Payment period not given
122        if paid:
123            # Only paid tickets are considered
124            for student in students:
125                for payment in student.get('payments', {}).values():
126                    if payment.p_state == 'paid':
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)
133    return payments
134
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
143
144class StudentExporterBase(ExporterBase):
145    """Exporter for students or related objects.
146    This is a baseclass.
147    """
148    grok.baseclass()
149    grok.implements(ICSVStudentExporter)
150    grok.provides(ICSVStudentExporter)
151
152    def filter_func(self, x, **kw):
153        return x
154
155    def get_filtered(self, site, **kw):
156        """Get students from a catalog filtered by keywords.
157        students_catalog is the default catalog. The keys must be valid
158        catalog index names.
159        Returns a simple empty list, a list with `Student`
160        objects or a catalog result set with `Student`
161        objects.
162
163        .. seealso:: `waeup.kofa.students.catalog.StudentsCatalog`
164
165        """
166        # Pass only given keywords to create FilteredCatalogQuery objects.
167        # This way we avoid
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.
172        if kw.get('catalog', None) == 'coursetickets':
173            coursetickets = CourseTicketsQuery(**kw).query()
174            students = []
175            for ticket in coursetickets:
176                students.append(ticket.student)
177            return list(set(students))
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
185        query = StudentsQuery(**kw)
186        return query.query()
187
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
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
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.
213        """
214        return self.export(self.filter_func(get_students(site)), filepath)
215
216    def export_student(self, student, filepath=None):
217        return self.export(self.filter_func([student]), filepath=filepath)
218
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)
225        return self.export(self.filter_func(data, **kw), filepath=filepath)
226
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)
233
234
235class StudentExporter(grok.GlobalUtility, StudentExporterBase):
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.
238    """
239    grok.name('students')
240
241    fields = tuple(sorted(iface_names(IStudent))) + (
242        'password', 'state', 'history', 'certcode', 'is_postgrad',
243        'current_level', 'current_session')
244    title = _(u'Students')
245
246    def mangle_value(self, value, name, context=None):
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."""
250        if name == 'history':
251            value = value.messages
252        if name == 'phone' and value is not None:
253            # Append hash '#' to phone numbers to circumvent
254            # unwanted excel automatic
255            value = str('%s#' % value)
256        return super(
257            StudentExporter, self).mangle_value(
258            value, name, context=context)
259
260
261class StudentStudyCourseExporter(grok.GlobalUtility, StudentExporterBase):
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.
266    """
267    grok.name('studentstudycourses')
268
269    fields = tuple(sorted(iface_names(IStudentStudyCourse))) + ('student_id',)
270    title = _(u'Student Study Courses')
271
272    def filter_func(self, x, **kw):
273        return get_studycourses(x)
274
275    def mangle_value(self, value, name, context=None):
276        """The mangler determines the certificate code and the student id.
277        """
278        if name == 'certificate' and value is not None:
279            # XXX: hopefully cert codes are unique site-wide
280            value = value.code
281        if name == 'student_id' and context is not None:
282            student = context.student
283            value = getattr(student, name, None)
284        return super(
285            StudentStudyCourseExporter, self).mangle_value(
286            value, name, context=context)
287
288
289class StudentStudyLevelExporter(grok.GlobalUtility, StudentExporterBase):
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
292    study level containers but not their content (course tickets).
293    The exporter iterates over all objects in the students' ``studycourse``
294    containers.
295    """
296    grok.name('studentstudylevels')
297
298    fields = tuple(sorted(iface_names(
299        IStudentStudyLevel) + ['level'])) + (
300        'student_id', 'number_of_tickets','certcode')
301    title = _(u'Student Study Levels')
302
303    def filter_func(self, x, **kw):
304        return get_levels(x)
305
306    def mangle_value(self, value, name, context=None):
307        """The mangler determines the student id, nothing else.
308        """
309        if name == 'student_id' and context is not None:
310            student = context.student
311            value = getattr(student, name, None)
312        return super(
313            StudentStudyLevelExporter, self).mangle_value(
314            value, name, context=context)
315
316class CourseTicketExporter(grok.GlobalUtility, StudentExporterBase):
317    """The Course Ticket  Exporter exports course tickets. Usually,
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
326    also meet level and level_session passed through at the same time.
327    This happens if the exporter is called at course level in the academics
328    section.
329    """
330    grok.name('coursetickets')
331
332    fields = tuple(sorted(iface_names(ICourseTicket) +
333        ['level', 'code', 'level_session'])) + ('student_id',
334        'certcode', 'display_fullname')
335    title = _(u'Course Tickets')
336
337    def filter_func(self, x, **kw):
338        return get_tickets(x, **kw)
339
340    def mangle_value(self, value, name, context=None):
341        """The mangler determines the student's id and fullname.
342        """
343        if context is not None:
344            student = context.student
345            if name in ('student_id', 'display_fullname') and student is not None:
346                value = getattr(student, name, None)
347        return super(
348            CourseTicketExporter, self).mangle_value(
349            value, name, context=context)
350
351
352class StudentPaymentsExporter(grok.GlobalUtility, StudentExporterBase):
353    """Exporter for OnlinePayment instances.
354    """
355    grok.name('studentpayments')
356
357    #: Fieldnames considered by this exporter
358    fields = tuple(
359        sorted(iface_names(
360            IStudentOnlinePayment, exclude_attribs=False,
361            omit=['display_item']))) + (
362            'student_id','state','current_session')
363
364    #: The title under which this exporter will be displayed
365    title = _(u'Student Payments')
366
367    def filter_func(self, x, **kw):
368        return get_payments(x, **kw)
369
370    def mangle_value(self, value, name, context=None):
371        """Treat location values special.
372        """
373        if context is not None:
374            student = context.student
375            if name in ['student_id','state',
376                        'current_session'] and student is not None:
377                value = getattr(student, name, None)
378        return super(
379            StudentPaymentsExporter, self).mangle_value(
380            value, name, context=context)
381
382class DataForBursaryExporter(StudentPaymentsExporter):
383    """Exporter for OnlinePayment instances.
384    """
385    grok.name('bursary')
386
387    def filter_func(self, x, **kw):
388        return get_payments(x, paid=True, **kw)
389
390    #: Fieldnames considered by this exporter
391    fields = tuple(
392        sorted(iface_names(
393            IStudentOnlinePayment, exclude_attribs=False,
394            omit=['display_item']))) + (
395            'student_id','matric_number','reg_number',
396            'firstname', 'middlename', 'lastname',
397            'state','current_session',
398            'entry_session', 'entry_mode',
399            'faccode', 'depcode','certcode')
400
401    #: The title under which this exporter will be displayed
402    title = _(u'Payment Data for Bursary')
403
404    def mangle_value(self, value, name, context=None):
405        """Treat location values special.
406        """
407        if context is not None:
408            student = context.student
409            if name in [
410                'student_id','matric_number', 'reg_number',
411                'firstname', 'middlename', 'lastname',
412                'state', 'current_session',
413                'entry_session', 'entry_mode',
414                'faccode', 'depcode', 'certcode'] and student is not None:
415                value = getattr(student, name, None)
416        return super(
417            StudentPaymentsExporter, self).mangle_value(
418            value, name, context=context)
419
420class BedTicketsExporter(grok.GlobalUtility, StudentExporterBase):
421    """Exporter for BedTicket instances.
422    """
423    grok.name('bedtickets')
424
425    #: Fieldnames considered by this exporter
426    fields = tuple(
427        sorted(iface_names(
428            IBedTicket, exclude_attribs=False,
429            omit=['display_coordinates']))) + (
430            'student_id', 'actual_bed_type')
431
432    #: The title under which this exporter will be displayed
433    title = _(u'Bed Tickets')
434
435    def filter_func(self, x, **kw):
436        return get_bedtickets(x)
437
438    def mangle_value(self, value, name, context=None):
439        """Treat location values and others special.
440        """
441        if context is not None:
442            student = context.student
443            if name in ['student_id'] and student is not None:
444                value = getattr(student, name, None)
445        if name == 'bed' and value is not None:
446            value = getattr(value, 'bed_id', None)
447        if name == 'actual_bed_type':
448            value = getattr(getattr(context, 'bed', None), 'bed_type')
449        return super(
450            BedTicketsExporter, self).mangle_value(
451            value, name, context=context)
452
453class StudentPaymentsOverviewExporter(StudentExporter):
454    """Exporter for students with payment overview.
455    """
456    grok.name('paymentsoverview')
457
458    curr_year = datetime.now().year
459    year_range = range(curr_year - 9, curr_year + 1)
460    year_range_tuple = tuple([str(year) for year in year_range])
461
462    #: Fieldnames considered by this exporter
463    fields = ('student_id', 'matric_number', 'display_fullname',
464        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
465        'current_level', 'current_session', 'current_mode',
466        ) + year_range_tuple
467
468    #: The title under which this exporter will be displayed
469    title = _(u'Student Payments Overview')
470
471    def mangle_value(self, value, name, context=None):
472        if name in self.year_range_tuple and context is not None:
473            value = 0
474            for ticket in context['payments'].values():
475                if ticket.p_category == 'schoolfee' and \
476                    ticket.p_session == int(name):
477                    if ticket.p_state == 'waived':
478                        value = 'waived'
479                        break
480                    if ticket.p_state == 'paid':
481                        try:
482                            value += ticket.amount_auth
483                        except TypeError:
484                            pass
485            if value == 0:
486                value = ''
487        return super(
488            StudentExporter, self).mangle_value(
489            value, name, context=context)
490
491class StudentStudyLevelsOverviewExporter(StudentExporter):
492    """Exporter for students with study level overview.
493    """
494    grok.name('studylevelsoverview')
495
496    avail_levels = tuple([str(x) for x in study_levels(None)])
497
498    #: Fieldnames considered by this exporter
499    fields = ('student_id', ) + (
500        'state', 'certcode', 'faccode', 'depcode', 'is_postgrad',
501        'entry_session', 'current_level', 'current_session',
502        ) + avail_levels
503
504    #: The title under which this exporter will be displayed
505    title = _(u'Student Study Levels Overview')
506
507    def mangle_value(self, value, name, context=None):
508        if name in self.avail_levels and context is not None:
509            value = ''
510            for level in context['studycourse'].values():
511                if level.level == int(name):
512                    #value = '%s|%s|%s|%s' % (
513                    #    level.level_session,
514                    #    len(level),
515                    #    level.validated_by,
516                    #    level.level_verdict)
517                    value = '%s' % level.level_session
518                    break
519        return super(
520            StudentExporter, self).mangle_value(
521            value, name, context=context)
522
523class ComboCardDataExporter(grok.GlobalUtility, StudentExporterBase):
524    """Exporter for Interswitch Combo Card Data.
525    """
526    grok.name('combocard')
527
528    #: Fieldnames considered by this exporter
529    fields = ('display_fullname',
530              'student_id','matric_number',
531              'certificate', 'faculty', 'department', 'passport_path')
532
533    #: The title under which this exporter will be displayed
534    title = _(u'Combo Card Data')
535
536    def mangle_value(self, value, name, context=None):
537        certificate = context['studycourse'].certificate
538        if name == 'certificate' and certificate is not None:
539            value = certificate.title
540        if name == 'department' and certificate is not None:
541            value = certificate.__parent__.__parent__.longtitle
542        if name == 'faculty' and certificate is not None:
543            value = certificate.__parent__.__parent__.__parent__.longtitle
544        if name == 'passport_path' and certificate is not None:
545            file_id = IFileStoreNameChooser(context).chooseName(attr='passport.jpg')
546            os_path = getUtility(IExtFileStore)._pathFromFileID(file_id)
547            if not os.path.exists(os_path):
548                value = None
549            else:
550                value = '/'.join(os_path.split('/')[-4:])
551        return super(
552            ComboCardDataExporter, self).mangle_value(
553            value, name, context=context)
Note: See TracBrowser for help on using the repository browser.