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

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

We need to import credits and passmark of course tickets in order to repair them. Let's use the form field validation for import.

Both passmark and credits must not be edited via the UI.

  • Property svn:keywords set to Id
File size: 9.4 KB
RevLine 
[8057]1## $Id: export.py 9316 2012-10-08 13:19:14Z 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) +
[9316]212        ['level', 'code', 'title',
213        '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))) + (
[9278]251                'student_id','student_state','current_session')
[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.