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

Last change on this file since 9272 was 9258, checked in by Henrik Bettermann, 12 years ago

Export payments with student_state to sort out if payments do not correspond with states set manually by officers.

  • Property svn:keywords set to Id
File size: 9.4 KB
RevLine 
[8057]1## $Id: export.py 9258 2012-09-30 13:05:04Z 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"""
20import grok
21from zope.catalog.interfaces import ICatalog
22from zope.component import queryUtility
23from waeup.kofa.interfaces import MessageFactory as _
[8015]24from waeup.kofa.students.interfaces import (
[8371]25    IStudent, IStudentStudyCourse, IStudentStudyLevel, ICourseTicket,
[8411]26    IStudentOnlinePayment, ICSVStudentExporter)
[7944]27from waeup.kofa.utils.batching import ExporterBase
28from waeup.kofa.utils.helpers import iface_names
29
[8400]30#: A tuple containing all exporter names referring to students or
31#: subobjects thereof.
32EXPORTER_NAMES = ('students', 'studentstudycourses', 'studentstudylevels',
33                  'coursetickets', 'studentpayments')
34
[8414]35def get_students(site):
36    """Get all students registered in catalog in `site`.
[7944]37    """
[8414]38    catalog = queryUtility(
39        ICatalog, context=site, name='students_catalog', default=None)
40    if catalog is None:
41        return []
42    students = catalog.searchResults(student_id=(None, None))
43    return students
44
45def get_studycourses(students):
46    """Get studycourses of `students`.
47    """
48    return [x.get('studycourse', None) for x in students
49            if x is not None]
50
51def get_levels(students):
52    """Get all studylevels of `students`.
53    """
54    levels = []
55    for course in get_studycourses(students):
56        for level in course.values():
57            levels.append(level)
58    return levels
59
60def get_tickets(students):
61    """Get all course tickets of `students`.
62    """
63    tickets = []
64    for level in get_levels(students):
65        for ticket in level.values():
66            tickets.append(ticket)
67    return tickets
68
69def get_payments(students):
70    """Get all payments of `students`.
71    """
72    payments = []
73    for student in students:
74        for payment in student.get('payments', {}).values():
75            payments.append(payment)
76    return payments
77
78
79class StudentExporterBase(ExporterBase):
80    """Exporter for students or related objects.
81
82    This is a baseclass.
83    """
84    grok.baseclass()
[8411]85    grok.implements(ICSVStudentExporter)
86    grok.provides(ICSVStudentExporter)
[8414]87
88    def export(self, values, filepath=None):
89        """Export `values`, an iterable, as CSV file.
90
91        If `filepath` is ``None``, a raw string with CSV data is returned.
92        """
93        writer, outfile = self.get_csv_writer(filepath)
94        for value in values:
95            self.write_item(value, writer)
96        return self.close_outfile(filepath, outfile)
97
98
99class StudentsExporter(grok.GlobalUtility, StudentExporterBase):
100    """Exporter for Students.
101    """
[7944]102    grok.name('students')
103
104    #: Fieldnames considered by this exporter
[8493]105    fields = tuple(sorted(iface_names(
106        IStudent, omit=['loggerInfo']))) + (
[9253]107        'password', 'state', 'history', 'certcode', 'is_postgrad',
108        'current_level', 'current_session')
[7944]109
110    #: The title under which this exporter will be displayed
111    title = _(u'Students')
112
[8493]113    def mangle_value(self, value, name, context=None):
114        if name == 'history':
115            value = value.messages
[8971]116        if name == 'phone' and value is not None:
117            # Append hash '#' to phone numbers to circumvent
118            # unwanted excel automatic
[8947]119            value = str('%s#' % value)
[8493]120        return super(
121            StudentsExporter, self).mangle_value(
122            value, name, context=context)
123
[7944]124    def export_all(self, site, filepath=None):
125        """Export students into filepath as CSV data.
126
127        If `filepath` is ``None``, a raw string with CSV data is returned.
128        """
[8414]129        return self.export(get_students(site), filepath)
[7994]130
[8411]131    def export_student(self, student, filepath=None):
132        return self.export([student], filepath=filepath)
133
[8414]134
135class StudentStudyCourseExporter(grok.GlobalUtility, StudentExporterBase):
[7994]136    """Exporter for StudentStudyCourses.
137    """
138    grok.name('studentstudycourses')
139
140    #: Fieldnames considered by this exporter
[8493]141    fields = tuple(sorted(iface_names(IStudentStudyCourse))) + ('student_id',)
[7994]142
143    #: The title under which this exporter will be displayed
144    title = _(u'Student Study Courses')
145
146    def mangle_value(self, value, name, context=None):
[8493]147        """Treat location values special.
[7994]148        """
149        if name == 'certificate' and value is not None:
150            # XXX: hopefully cert codes are unique site-wide
151            value = value.code
[8493]152        if name == 'student_id' and context is not None:
[8736]153            student = context.student
[8493]154            value = getattr(student, name, None)
[7994]155        return super(
156            StudentStudyCourseExporter, self).mangle_value(
157            value, name, context=context)
158
159    def export_all(self, site, filepath=None):
160        """Export study courses into filepath as CSV data.
161
162        If `filepath` is ``None``, a raw string with CSV data is returned.
163        """
[8414]164        return self.export(get_studycourses(get_students(site)), filepath)
[8015]165
[8411]166    def export_student(self, student, filepath=None):
[8414]167        """Export studycourse of a single student object.
168        """
169        return self.export(get_studycourses([student]), filepath)
[8411]170
171
[8414]172class StudentStudyLevelExporter(grok.GlobalUtility, StudentExporterBase):
[8015]173    """Exporter for StudentStudyLevels.
174    """
175    grok.name('studentstudylevels')
176
177    #: Fieldnames considered by this exporter
[8493]178    fields = tuple(sorted(iface_names(
[9253]179        IStudentStudyLevel) + ['level'])) + (
180        'student_id', 'number_of_tickets','certcode')
[8015]181
182    #: The title under which this exporter will be displayed
183    title = _(u'Student Study Levels')
184
185    def mangle_value(self, value, name, context=None):
[8493]186        """Treat location values special.
[8015]187        """
[8493]188        if name == 'student_id' and context is not None:
[8736]189            student = context.student
[8493]190            value = getattr(student, name, None)
[8015]191        return super(
192            StudentStudyLevelExporter, self).mangle_value(
193            value, name, context=context)
194
195    def export_all(self, site, filepath=None):
196        """Export study levels into filepath as CSV data.
197
198        If `filepath` is ``None``, a raw string with CSV data is returned.
199        """
[8414]200        return self.export(get_levels(get_students(site)), filepath)
[8342]201
[8411]202    def export_student(self, student, filepath=None):
[8414]203        return self.export(get_levels([student]), filepath)
[8411]204
[8414]205class CourseTicketExporter(grok.GlobalUtility, StudentExporterBase):
[8342]206    """Exporter for CourseTickets.
207    """
208    grok.name('coursetickets')
209
210    #: Fieldnames considered by this exporter
[8493]211    fields = tuple(sorted(iface_names(ICourseTicket) +
212        ['level', 'code', 'title', 'credits',
[9253]213        'passmark', 'semester', 'fcode', 'dcode'])) + ('student_id', 'certcode')
[8342]214
215    #: The title under which this exporter will be displayed
216    title = _(u'Course Tickets')
217
218    def mangle_value(self, value, name, context=None):
219        """Treat location values special.
220        """
221        if context is not None:
[8736]222            student = context.student
[8493]223            if name == 'student_id' and student is not None:
[8342]224                value = getattr(student, name, None)
225            if name == 'level':
[8393]226                value = getattr(context, 'getLevel', lambda: None)()
[8342]227        return super(
228            CourseTicketExporter, self).mangle_value(
229            value, name, context=context)
230
231    def export_all(self, site, filepath=None):
232        """Export course tickets into filepath as CSV data.
233
234        If `filepath` is ``None``, a raw string with CSV data is returned.
235        """
[8414]236        return self.export(get_tickets(get_students(site)), filepath)
[8342]237
[8411]238    def export_student(self, student, filepath=None):
[8414]239        return self.export(get_tickets([student]), filepath)
[8411]240
[8414]241
242class PaymentsExporter(grok.GlobalUtility, StudentExporterBase):
[8371]243    """Exporter for OnlinePayment instances.
244    """
245    grok.name('studentpayments')
246
247    #: Fieldnames considered by this exporter
248    fields = tuple(
[8493]249        sorted(iface_names(
[9258]250            IStudentOnlinePayment, exclude_attribs=False))) + (
251                'student_id','student_state')
[8371]252
253    #: The title under which this exporter will be displayed
[8576]254    title = _(u'Student Payments')
[8371]255
256    def mangle_value(self, value, name, context=None):
257        """Treat location values special.
258        """
259        if context is not None:
[8736]260            student = context.student
[8493]261            if name in ['student_id'] and student is not None:
[8371]262                value = getattr(student, name, None)
263        return super(
264            PaymentsExporter, self).mangle_value(
265            value, name, context=context)
266
267    def export_all(self, site, filepath=None):
268        """Export payments into filepath as CSV data.
269
270        If `filepath` is ``None``, a raw string with CSV data is returned.
271        """
[8414]272        return self.export(get_payments(get_students(site)), filepath)
[8411]273
274    def export_student(self, student, filepath=None):
[8414]275        return self.export(get_payments([student]), filepath)
Note: See TracBrowser for help on using the repository browser.