source: main/waeup.kofa/trunk/src/waeup/kofa/students/batching.py @ 9721

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

Rework logging of batch processing. Remove redundant text but add name of import file.

  • Property svn:keywords set to Id
File size: 31.9 KB
RevLine 
[7191]1## $Id: batching.py 9706 2012-11-21 22:37:03Z 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##
[7433]18"""Batch processing components for student objects.
[6821]19
20Batch processors eat CSV files to add, update or remove large numbers
21of certain kinds of objects at once.
22
[7261]23Here we define the processors for students specific objects like
24students, studycourses, payment tickets and accommodation tickets.
[6821]25"""
26import grok
[6849]27import csv
[8884]28from time import time
[9295]29from datetime import datetime
[9296]30from zope.i18n import translate
[6821]31from zope.interface import Interface
[6825]32from zope.schema import getFields
[7548]33from zope.component import queryUtility, getUtility
[7429]34from zope.event import notify
[6825]35from zope.catalog.interfaces import ICatalog
[7951]36from hurry.workflow.interfaces import IWorkflowState, IWorkflowInfo
[7811]37from waeup.kofa.interfaces import (
[7522]38    IBatchProcessor, FatalCSVError, IObjectConverter, IUserAccount,
[9284]39    IObjectHistory, VALIDATED, REGISTERED, IGNORE_MARKER)
[9296]40from waeup.kofa.interfaces import IKofaUtils
[7959]41from waeup.kofa.interfaces import MessageFactory as _
[7811]42from waeup.kofa.students.interfaces import (
[7532]43    IStudent, IStudentStudyCourse,
[7536]44    IStudentUpdateByRegNo, IStudentUpdateByMatricNo,
[9420]45    IStudentStudyLevel, ICourseTicketImport,
[8174]46    IStudentOnlinePayment, IStudentVerdictUpdate)
[8309]47from waeup.kofa.students.workflow import  (
[9028]48    IMPORTABLE_STATES, IMPORTABLE_TRANSITIONS,
49    FORBIDDEN_POSTGRAD_TRANS, FORBIDDEN_POSTGRAD_STATES)
[7811]50from waeup.kofa.utils.batching import BatchProcessor
[6821]51
52class StudentProcessor(BatchProcessor):
53    """A batch processor for IStudent objects.
54    """
55    grok.implements(IBatchProcessor)
56    grok.provides(IBatchProcessor)
57    grok.context(Interface)
[7933]58    util_name = 'studentprocessor'
[6821]59    grok.name(util_name)
60
[7933]61    name = u'Student Processor'
[6821]62    iface = IStudent
[8581]63    iface_byregnumber = IStudentUpdateByRegNo
64    iface_bymatricnumber = IStudentUpdateByMatricNo
[6821]65
[6849]66    location_fields = []
[6821]67    factory_name = 'waeup.Student'
68
69    @property
[6849]70    def available_fields(self):
[8176]71        fields = getFields(self.iface)
[6849]72        return sorted(list(set(
[7513]73            ['student_id','reg_number','matric_number',
[8309]74            'password', 'state', 'transition'] + fields.keys())))
[6821]75
[6849]76    def checkHeaders(self, headerfields, mode='create'):
[8309]77        if 'state' in headerfields and 'transition' in headerfields:
78            raise FatalCSVError(
79                "State and transition can't be  imported at the same time!")
[6854]80        if not 'reg_number' in headerfields and not 'student_id' \
81            in headerfields and not 'matric_number' in headerfields:
[6849]82            raise FatalCSVError(
[6854]83                "Need at least columns student_id or reg_number " +
84                "or matric_number for import!")
[6849]85        if mode == 'create':
86            for field in self.required_fields:
87                if not field in headerfields:
88                    raise FatalCSVError(
89                        "Need at least columns %s for import!" %
90                        ', '.join(["'%s'" % x for x in self.required_fields]))
91        # Check for fields to be ignored...
92        not_ignored_fields = [x for x in headerfields
93                              if not x.startswith('--')]
94        if len(set(not_ignored_fields)) < len(not_ignored_fields):
95            raise FatalCSVError(
96                "Double headers: each column name may only appear once.")
97        return True
98
[6821]99    def parentsExist(self, row, site):
100        return 'students' in site.keys()
101
[6849]102    def getLocator(self, row):
[8232]103        if row.get('student_id',None) not in (None, IGNORE_MARKER):
[6849]104            return 'student_id'
[8232]105        elif row.get('reg_number',None) not in (None, IGNORE_MARKER):
[6849]106            return 'reg_number'
[8232]107        elif row.get('matric_number',None) not in (None, IGNORE_MARKER):
[6849]108            return 'matric_number'
109        else:
110            return None
111
[6821]112    # The entry never exists in create mode.
113    def entryExists(self, row, site):
[7267]114        return self.getEntry(row, site) is not None
115
116    def getParent(self, row, site):
117        return site['students']
118
119    def getEntry(self, row, site):
[6846]120        if not 'students' in site.keys():
[6849]121            return None
122        if self.getLocator(row) == 'student_id':
[6846]123            if row['student_id'] in site['students']:
124                student = site['students'][row['student_id']]
125                return student
[6849]126        elif self.getLocator(row) == 'reg_number':
[6846]127            reg_number = row['reg_number']
128            cat = queryUtility(ICatalog, name='students_catalog')
129            results = list(
130                cat.searchResults(reg_number=(reg_number, reg_number)))
131            if results:
132                return results[0]
[6849]133        elif self.getLocator(row) == 'matric_number':
[6846]134            matric_number = row['matric_number']
135            cat = queryUtility(ICatalog, name='students_catalog')
136            results = list(
137                cat.searchResults(matric_number=(matric_number, matric_number)))
138            if results:
139                return results[0]
[6849]140        return None
[6821]141
142    def addEntry(self, obj, row, site):
143        parent = self.getParent(row, site)
144        parent.addStudent(obj)
[8491]145        # Reset _curr_stud_id if student_id has been imported
146        if self.getLocator(row) == 'student_id':
147            parent._curr_stud_id -= 1
[8287]148        # We have to log this if state is provided. If not,
[7959]149        # logging is done by the event handler handle_student_added
[9701]150        if 'state' in row:
[7959]151            parent.logger.info('%s - Student record created' % obj.student_id)
[7656]152        history = IObjectHistory(obj)
[7959]153        history.addMessage(_('Student record created'))
[6821]154        return
155
156    def delEntry(self, row, site):
[7267]157        student = self.getEntry(row, site)
[7263]158        if student is not None:
[6846]159            parent = self.getParent(row, site)
[7656]160            parent.logger.info('%s - Student removed' % student.student_id)
[6846]161            del parent[student.student_id]
[6821]162        pass
[6825]163
[8309]164    def checkUpdateRequirements(self, obj, row, site):
165        """Checks requirements the object must fulfill when being updated.
166
167        This method is not used in case of deleting or adding objects.
168
169        Returns error messages as strings in case of requirement
170        problems.
171        """
172        transition = row.get('transition', IGNORE_MARKER)
173        if transition not in (IGNORE_MARKER, ''):
174            allowed_transitions = IWorkflowInfo(obj).getManualTransitionIds()
175            if transition not in allowed_transitions:
176                return 'Transition not allowed.'
[9028]177            if transition in FORBIDDEN_POSTGRAD_TRANS and \
178                obj.is_postgrad:
179                return 'Transition not allowed (pg student).'
180        state = row.get('state', IGNORE_MARKER)
181        if state not in (IGNORE_MARKER, ''):
182            if state in FORBIDDEN_POSTGRAD_STATES and \
183                obj.is_postgrad:
184                return 'State not allowed (pg student).'
[8309]185        return None
186
[9706]187    def updateEntry(self, obj, row, site, filename):
[7497]188        """Update obj to the values given in row.
189        """
[8221]190        items_changed = ''
191
[7643]192        # Remove student_id from row if empty
[9701]193        if 'student_id' in row and row['student_id'] in (None, IGNORE_MARKER):
[7643]194            row.pop('student_id')
[8221]195
196        # Update password
[9701]197        # XXX: Take DELETION_MARKER into consideration
198        if 'password' in row:
[8498]199            passwd = row.get('password', IGNORE_MARKER)
200            if passwd not in ('', IGNORE_MARKER):
201                if passwd.startswith('{SSHA}'):
202                    # already encrypted password
203                    obj.password = passwd
204                else:
205                    # not yet encrypted password
206                    IUserAccount(obj).setPassword(passwd)
207                items_changed += ('%s=%s, ' % ('password', passwd))
[8221]208            row.pop('password')
209
210        # Update registration state
[9701]211        if 'state' in row:
[8498]212            state = row.get('state', IGNORE_MARKER)
213            if state not in (IGNORE_MARKER, ''):
214                value = row['state']
215                IWorkflowState(obj).setState(value)
216                msg = _("State '${a}' set", mapping = {'a':value})
217                history = IObjectHistory(obj)
218                history.addMessage(msg)
219                items_changed += ('%s=%s, ' % ('state', state))
[8287]220            row.pop('state')
[8498]221
[9701]222        if 'transition' in row:
[8498]223            transition = row.get('transition', IGNORE_MARKER)
224            if transition not in (IGNORE_MARKER, ''):
225                value = row['transition']
226                IWorkflowInfo(obj).fireTransition(value)
227                items_changed += ('%s=%s, ' % ('transition', transition))
[8309]228            row.pop('transition')
[8221]229
230        # apply other values...
[8498]231        items_changed += super(StudentProcessor, self).updateEntry(
[9706]232            obj, row, site, filename)
[8221]233
234        # Log actions...
[7656]235        parent = self.getParent(row, site)
236        if hasattr(obj,'student_id'):
[8995]237            # Update mode: the student exists and we can get the student_id.
238            # Create mode: the record contains the student_id
[7656]239            parent.logger.info(
[9706]240                '%s - %s - %s - updated: %s'
241                % (self.name, filename, obj.student_id, items_changed))
[7656]242        else:
243            # Create mode: the student does not yet exist
[9706]244            # XXX: It seems that this never happens because student_id
245            # is always set.
246            parent.logger.info(
247                '%s - %s - %s - imported: %s'
248                % (self.name, filename, obj.student_id, items_changed))
[8221]249        return items_changed
[7497]250
[6849]251    def getMapping(self, path, headerfields, mode):
252        """Get a mapping from CSV file headerfields to actually used fieldnames.
253        """
254        result = dict()
255        reader = csv.reader(open(path, 'rb'))
256        raw_header = reader.next()
257        for num, field in enumerate(headerfields):
[8221]258            if field not in ['student_id', 'reg_number', 'matric_number'
259                             ] and mode == 'remove':
[6849]260                continue
261            if field == u'--IGNORE--':
262                # Skip ignored columns in failed and finished data files.
263                continue
264            result[raw_header[num]] = field
265        return result
266
267    def checkConversion(self, row, mode='create'):
268        """Validates all values in row.
269        """
[7643]270        iface = self.iface
[6849]271        if mode in ['update', 'remove']:
272            if self.getLocator(row) == 'reg_number':
[8581]273                iface = self.iface_byregnumber
[6849]274            elif self.getLocator(row) == 'matric_number':
[8581]275                iface = self.iface_bymatricnumber
[6849]276        converter = IObjectConverter(iface)
277        errs, inv_errs, conv_dict =  converter.fromStringDict(
[8214]278            row, self.factory_name, mode=mode)
[9701]279        if 'transition' in row:
[9028]280            if row['transition'] not in IMPORTABLE_TRANSITIONS:
281                if row['transition'] not in (IGNORE_MARKER, ''):
282                    errs.append(('transition','not allowed'))
[9701]283        if 'state' in row:
[9028]284            if row['state'] not in IMPORTABLE_STATES:
285                if row['state'] not in (IGNORE_MARKER, ''):
286                    errs.append(('state','not allowed'))
287                else:
288                    # State is an attribute of Student and must not
289                    # be changed if empty.
290                    conv_dict['state'] = IGNORE_MARKER
[8490]291        try:
292            # Correct stud_id counter. As the IConverter for students
293            # creates student objects that are not used afterwards, we
294            # have to fix the site-wide student_id counter.
295            site = grok.getSite()
296            students = site['students']
297            students._curr_stud_id -= 1
298        except (KeyError, TypeError, AttributeError):
299                pass
[6849]300        return errs, inv_errs, conv_dict
301
[8232]302
303class StudentProcessorBase(BatchProcessor):
304    """A base for student subitem processor.
305
306    Helps reducing redundancy.
[6825]307    """
[8232]308    grok.baseclass()
[6825]309
[9420]310    # additional available fields
[8884]311    # beside 'student_id', 'reg_number' and 'matric_number'
[8232]312    additional_fields = []
[6825]313
[8884]314    #: header fields additionally required
[8232]315    additional_headers = []
[6849]316
[6825]317    @property
318    def available_fields(self):
[8232]319        fields = ['student_id','reg_number','matric_number'
320                  ] + self.additional_fields
321        return sorted(list(set(fields + getFields(
[6843]322                self.iface).keys())))
[6825]323
[6837]324    def checkHeaders(self, headerfields, mode='ignore'):
[6854]325        if not 'reg_number' in headerfields and not 'student_id' \
326            in headerfields and not 'matric_number' in headerfields:
[6825]327            raise FatalCSVError(
[6854]328                "Need at least columns student_id " +
329                "or reg_number or matric_number for import!")
[8232]330        for name in self.additional_headers:
331            if not name in headerfields:
332                raise FatalCSVError(
333                    "Need %s for import!" % name)
334
[6834]335        # Check for fields to be ignored...
[6825]336        not_ignored_fields = [x for x in headerfields
337                              if not x.startswith('--')]
338        if len(set(not_ignored_fields)) < len(not_ignored_fields):
339            raise FatalCSVError(
340                "Double headers: each column name may only appear once.")
341        return True
342
[8232]343    def _getStudent(self, row, site):
[8225]344        NON_VALUES = ['', IGNORE_MARKER]
[6846]345        if not 'students' in site.keys():
[6849]346            return None
[8225]347        if row.get('student_id', '') not in NON_VALUES:
[6825]348            if row['student_id'] in site['students']:
349                student = site['students'][row['student_id']]
350                return student
[8225]351        elif row.get('reg_number', '') not in NON_VALUES:
[6825]352            reg_number = row['reg_number']
353            cat = queryUtility(ICatalog, name='students_catalog')
354            results = list(
355                cat.searchResults(reg_number=(reg_number, reg_number)))
356            if results:
357                return results[0]
[8225]358        elif row.get('matric_number', '') not in NON_VALUES:
[6843]359            matric_number = row['matric_number']
360            cat = queryUtility(ICatalog, name='students_catalog')
361            results = list(
362                cat.searchResults(matric_number=(matric_number, matric_number)))
363            if results:
364                return results[0]
[6849]365        return None
[6825]366
[7267]367    def parentsExist(self, row, site):
368        return self.getParent(row, site) is not None
369
[6825]370    def entryExists(self, row, site):
[7534]371        return self.getEntry(row, site) is not None
[6825]372
[8232]373    def checkConversion(self, row, mode='ignore'):
374        """Validates all values in row.
375        """
376        converter = IObjectConverter(self.iface)
377        errs, inv_errs, conv_dict =  converter.fromStringDict(
378            row, self.factory_name, mode=mode)
379        return errs, inv_errs, conv_dict
380
[8885]381    def getMapping(self, path, headerfields, mode):
382        """Get a mapping from CSV file headerfields to actually used fieldnames.
383        """
384        result = dict()
385        reader = csv.reader(open(path, 'rb'))
386        raw_header = reader.next()
387        for num, field in enumerate(headerfields):
[8888]388            if field not in ['student_id', 'reg_number', 'matric_number',
389                             'p_id', 'code', 'level'
[8885]390                             ] and mode == 'remove':
391                continue
392            if field == u'--IGNORE--':
393                # Skip ignored columns in failed and finished data files.
394                continue
395            result[raw_header[num]] = field
396        return result
[8232]397
[8885]398
[8232]399class StudentStudyCourseProcessor(StudentProcessorBase):
400    """A batch processor for IStudentStudyCourse objects.
401    """
402    grok.implements(IBatchProcessor)
403    grok.provides(IBatchProcessor)
404    grok.context(Interface)
405    util_name = 'studycourseupdater'
406    grok.name(util_name)
407
408    name = u'StudentStudyCourse Processor (update only)'
409    iface = IStudentStudyCourse
410    factory_name = 'waeup.StudentStudyCourse'
411
412    location_fields = []
413    additional_fields = []
414
415    def getParent(self, row, site):
416        return self._getStudent(row, site)
417
[6825]418    def getEntry(self, row, site):
[7534]419        student = self.getParent(row, site)
[7536]420        if student is None:
[6825]421            return None
422        return student.get('studycourse')
[7429]423
[9706]424    def updateEntry(self, obj, row, site, filename):
[7429]425        """Update obj to the values given in row.
426        """
[8221]427        items_changed = super(StudentStudyCourseProcessor, self).updateEntry(
[9706]428            obj, row, site, filename)
[7656]429        parent = self.getParent(row, site)
430        parent.__parent__.logger.info(
[9706]431            '%s - %s - %s - updated: %s'
432            % (self.name, filename, parent.student_id, items_changed))
[7429]433        # Update the students_catalog
434        notify(grok.ObjectModifiedEvent(obj.__parent__))
435        return
436
[7532]437    def checkConversion(self, row, mode='ignore'):
438        """Validates all values in row.
439        """
[8232]440        errs, inv_errs, conv_dict = super(
441            StudentStudyCourseProcessor, self).checkConversion(row, mode=mode)
[7532]442        # We have to check if current_level is in range of certificate.
[9701]443        if 'certificate' in conv_dict and 'current_level' in conv_dict:
[8940]444            cert = conv_dict['certificate']
445            level = conv_dict['current_level']
446            if level < cert.start_level or level > cert.end_level+120:
447                errs.append(('current_level','not in range'))
[7532]448        return errs, inv_errs, conv_dict
449
[9028]450    def checkUpdateRequirements(self, obj, row, site):
451        """Checks requirements the object must fulfill when being updated.
452
453        Returns error messages as strings in case of requirement
454        problems.
455        """
456        current_level = row.get('current_level', None)
[9037]457        if current_level == 999 and \
[9029]458            obj.__parent__.state in FORBIDDEN_POSTGRAD_STATES:
[9028]459            return 'Not a pg student.'
460        return None
461
[8232]462class StudentStudyLevelProcessor(StudentProcessorBase):
[7536]463    """A batch processor for IStudentStudyLevel objects.
464    """
465    grok.implements(IBatchProcessor)
466    grok.provides(IBatchProcessor)
467    grok.context(Interface)
[7933]468    util_name = 'studylevelprocessor'
[7536]469    grok.name(util_name)
470
[7933]471    name = u'StudentStudyLevel Processor'
[7536]472    iface = IStudentStudyLevel
473    factory_name = 'waeup.StudentStudyLevel'
474
475    location_fields = []
[9160]476
[9161]477    additional_fields = ['level']
[8232]478    additional_headers = ['level']
[7536]479
[9690]480    @property
481    def available_fields(self):
482        fields = super(StudentStudyLevelProcessor, self).available_fields
483        fields.remove('total_credits')
484        fields.remove('gpa')
485        return  fields
486
[7536]487    def getParent(self, row, site):
[8232]488        student = self._getStudent(row, site)
489        if student is None:
[7536]490            return None
[8232]491        return student['studycourse']
[7536]492
493    def getEntry(self, row, site):
494        studycourse = self.getParent(row, site)
495        if studycourse is None:
496            return None
497        return studycourse.get(row['level'])
498
[9301]499    def delEntry(self, row, site):
500        studylevel = self.getEntry(row, site)
501        parent = self.getParent(row, site)
502        if studylevel is not None:
503            student = self._getStudent(row, site)
504            student.__parent__.logger.info('%s - Level removed: %s'
505                % (student.student_id, studylevel.__name__))
506            del parent[studylevel.__name__]
507        return
508
[9706]509    def updateEntry(self, obj, row, site, filename):
[8626]510        """Update obj to the values given in row.
511        """
512        items_changed = super(StudentStudyLevelProcessor, self).updateEntry(
[9706]513            obj, row, site, filename)
[8626]514        student = self.getParent(row, site).__parent__
515        student.__parent__.logger.info(
[9706]516            '%s - %s - %s - updated: %s'
517            % (self.name, filename, student.student_id, items_changed))
[8626]518        return
519
[7536]520    def addEntry(self, obj, row, site):
521        parent = self.getParent(row, site)
522        obj.level = int(row['level'])
523        parent[row['level']] = obj
524        return
525
526    def checkConversion(self, row, mode='ignore'):
527        """Validates all values in row.
528        """
[8232]529        errs, inv_errs, conv_dict = super(
530            StudentStudyLevelProcessor, self).checkConversion(row, mode=mode)
531
[7536]532        # We have to check if level is a valid integer.
[7548]533        # This is not done by the converter.
[7536]534        try:
535            level = int(row['level'])
[9301]536            if level not in range(0,1000,10) + [999]:
[7536]537                errs.append(('level','no valid integer'))
538        except ValueError:
539            errs.append(('level','no integer'))
540        return errs, inv_errs, conv_dict
[7548]541
[8232]542class CourseTicketProcessor(StudentProcessorBase):
[7548]543    """A batch processor for ICourseTicket objects.
544    """
545    grok.implements(IBatchProcessor)
546    grok.provides(IBatchProcessor)
547    grok.context(Interface)
[7933]548    util_name = 'courseticketprocessor'
[7548]549    grok.name(util_name)
550
[7933]551    name = u'CourseTicket Processor'
[9420]552    iface = ICourseTicketImport
[7548]553    factory_name = 'waeup.CourseTicket'
554
555    location_fields = []
[9316]556    additional_fields = ['level', 'code']
[8232]557    additional_headers = ['level', 'code']
[7548]558
[9420]559    @property
560    def available_fields(self):
561        fields = [
562            'student_id','reg_number','matric_number',
563            'mandatory', 'score', 'carry_over', 'automatic',
564            'level_session'
565            ] + self.additional_fields
566        return sorted(fields)
567
[7548]568    def getParent(self, row, site):
[8232]569        student = self._getStudent(row, site)
570        if student is None:
[7548]571            return None
[8232]572        return student['studycourse'].get(row['level'])
[7548]573
574    def getEntry(self, row, site):
575        level = self.getParent(row, site)
576        if level is None:
577            return None
578        return level.get(row['code'])
579
[9706]580    def updateEntry(self, obj, row, site, filename):
[8626]581        """Update obj to the values given in row.
582        """
583        items_changed = super(CourseTicketProcessor, self).updateEntry(
[9706]584            obj, row, site, filename)
[8888]585        parent = self.getParent(row, site)
[8626]586        student = self.getParent(row, site).__parent__.__parent__
587        student.__parent__.logger.info(
[9706]588            '%s - %s - %s - %s - updated: %s'
589            % (self.name, filename, student.student_id, parent.level, items_changed))
[8626]590        return
591
[7548]592    def addEntry(self, obj, row, site):
593        parent = self.getParent(row, site)
594        catalog = getUtility(ICatalog, name='courses_catalog')
595        entries = list(catalog.searchResults(code=(row['code'],row['code'])))
596        obj.fcode = entries[0].__parent__.__parent__.__parent__.code
597        obj.dcode = entries[0].__parent__.__parent__.code
598        obj.title = entries[0].title
[9420]599        #if getattr(obj, 'credits', None) is None:
600        obj.credits = entries[0].credits
601        #if getattr(obj, 'passmark', None) is None:
602        obj.passmark = entries[0].passmark
[7548]603        obj.semester = entries[0].semester
604        parent[row['code']] = obj
605        return
606
[8888]607    def delEntry(self, row, site):
608        ticket = self.getEntry(row, site)
609        parent = self.getParent(row, site)
610        if ticket is not None:
611            student = self._getStudent(row, site)
612            student.__parent__.logger.info('%s - Course ticket in %s removed: %s'
613                % (student.student_id, parent.level, ticket.code))
614            del parent[ticket.code]
615        return
616
[7548]617    def checkConversion(self, row, mode='ignore'):
618        """Validates all values in row.
619        """
[8232]620        errs, inv_errs, conv_dict = super(
621            CourseTicketProcessor, self).checkConversion(row, mode=mode)
[9420]622        if mode == 'remove':
623            return errs, inv_errs, conv_dict
624        # In update and create mode we have to check if course really exists.
[7548]625        # This is not done by the converter.
626        catalog = getUtility(ICatalog, name='courses_catalog')
627        entries = catalog.searchResults(code=(row['code'],row['code']))
628        if len(entries) == 0:
629            errs.append(('code','non-existent'))
630            return errs, inv_errs, conv_dict
[9420]631        # If level_session is provided in row we have to check if
632        # the parent studylevel exists and if its level_session
633        # attribute corresponds with the expected value in row.
[9421]634        level_session = conv_dict.get('level_session', IGNORE_MARKER)
635        if level_session not in (IGNORE_MARKER, None):
[9420]636            site = grok.getSite()
637            studylevel = self.getParent(row, site)
638            if studylevel is not None:
639                if studylevel.level_session != level_session:
640                    errs.append(('level_session','does not match %s'
641                        % studylevel.level_session))
642            else:
643                errs.append(('level','does not exist'))
[7623]644        return errs, inv_errs, conv_dict
645
[8232]646class StudentOnlinePaymentProcessor(StudentProcessorBase):
[7623]647    """A batch processor for IStudentOnlinePayment objects.
648    """
649    grok.implements(IBatchProcessor)
650    grok.provides(IBatchProcessor)
651    grok.context(Interface)
[7933]652    util_name = 'paymentprocessor'
[7623]653    grok.name(util_name)
654
[8942]655    name = u'Student Payment Processor'
[8174]656    iface = IStudentOnlinePayment
[7623]657    factory_name = 'waeup.StudentOnlinePayment'
658
659    location_fields = []
[8232]660    additional_fields = ['p_id']
[8884]661    additional_headers = []
[7623]662
[8884]663    def checkHeaders(self, headerfields, mode='ignore'):
664        super(StudentOnlinePaymentProcessor, self).checkHeaders(headerfields)
[8885]665        if mode in ('update', 'remove') and not 'p_id' in headerfields:
[8884]666            raise FatalCSVError(
[8885]667                "Need p_id for import in update and remove modes!")
[8884]668        return True
669
[8232]670    def parentsExist(self, row, site):
671        return self.getParent(row, site) is not None
[7623]672
673    def getParent(self, row, site):
[8232]674        student = self._getStudent(row, site)
675        if student is None:
[7623]676            return None
[8232]677        return student['payments']
[7623]678
679    def getEntry(self, row, site):
680        payments = self.getParent(row, site)
681        if payments is None:
682            return None
[8884]683        p_id = row.get('p_id', None)
684        if p_id is None:
685            return None
[7626]686        # We can use the hash symbol at the end of p_id in import files
687        # to avoid annoying automatic number transformation
688        # by Excel or Calc
[8884]689        p_id = p_id.strip('#')
[9467]690        if len(p_id.split('-')) != 3 and not p_id.startswith('p'):
[8884]691            # For data migration from old SRP only
692            p_id = 'p' + p_id[7:] + '0'
693        entry = payments.get(p_id)
[7623]694        return entry
695
[9706]696    def updateEntry(self, obj, row, site, filename):
[8626]697        """Update obj to the values given in row.
698        """
699        items_changed = super(StudentOnlinePaymentProcessor, self).updateEntry(
[9706]700            obj, row, site, filename)
[8626]701        student = self.getParent(row, site).__parent__
702        student.__parent__.logger.info(
[9706]703            '%s - %s - %s - updated: %s'
704            % (self.name, filename, student.student_id, items_changed))
[8626]705        return
706
[7623]707    def addEntry(self, obj, row, site):
708        parent = self.getParent(row, site)
[7626]709        p_id = row['p_id'].strip('#')
[9467]710        if len(p_id.split('-')) != 3 and not p_id.startswith('p'):
[7623]711            # For data migration from old SRP
[8884]712            obj.p_id = 'p' + p_id[7:] + '0'
[7623]713            parent[obj.p_id] = obj
714        else:
[7626]715            parent[p_id] = obj
[7623]716        return
717
[8885]718    def delEntry(self, row, site):
719        payment = self.getEntry(row, site)
720        parent = self.getParent(row, site)
721        if payment is not None:
722            student = self._getStudent(row, site)
[8886]723            student.__parent__.logger.info('%s - Payment ticket removed: %s'
[8885]724                % (student.student_id, payment.p_id))
725            del parent[payment.p_id]
[8886]726        return
[8885]727
[7623]728    def checkConversion(self, row, mode='ignore'):
729        """Validates all values in row.
730        """
[8232]731        errs, inv_errs, conv_dict = super(
732            StudentOnlinePaymentProcessor, self).checkConversion(row, mode=mode)
733
[7623]734        # We have to check p_id.
[8884]735        p_id = row.get('p_id', None)
[8942]736        if not p_id:
[8884]737            timestamp = ("%d" % int(time()*10000))[1:]
738            p_id = "p%s" % timestamp
739            conv_dict['p_id'] = p_id
740            return errs, inv_errs, conv_dict
741        else:
742            p_id = p_id.strip('#')
[7626]743        if p_id.startswith('p'):
744            if not len(p_id) == 14:
[7623]745                errs.append(('p_id','invalid length'))
746                return errs, inv_errs, conv_dict
[9467]747        elif len(p_id.split('-')) == 3:
748            # The SRP used either pins as keys ...
[9479]749            if len(p_id.split('-')[2]) not in (9, 10):
[9467]750                errs.append(('p_id','invalid pin'))
751                return errs, inv_errs, conv_dict
[7623]752        else:
[9467]753            # ... or order_ids.
[7626]754            if not len(p_id) == 19:
[7623]755                errs.append(('p_id','invalid length'))
756                return errs, inv_errs, conv_dict
757        return errs, inv_errs, conv_dict
[7951]758
759class StudentVerdictProcessor(StudentStudyCourseProcessor):
[9418]760    """A special batch processor for verdicts.
[7951]761
762    Import verdicts and perform workflow transitions.
763    """
764
765    util_name = 'verdictupdater'
766    grok.name(util_name)
767
[9418]768    name = u'Verdict Processor (special processor, update only)'
[7951]769    iface = IStudentVerdictUpdate
770    factory_name = 'waeup.StudentStudyCourse'
771
[9284]772    additional_fields = [
773        'current_session',
774        'current_level',
[9296]775        'bypass_validation',
776        'validated_by']
[7951]777
778    def checkUpdateRequirements(self, obj, row, site):
779        """Checks requirements the studycourse and the student must fulfill
780        before being updated.
781        """
782        # Check if current_levels correspond
783        if obj.current_level != row['current_level']:
784            return 'Current level does not correspond.'
785        # Check if current_sessions correspond
786        if obj.current_session != row['current_session']:
787            return 'Current session does not correspond.'
[9282]788        # Check if new verdict is provided
789        if row['current_verdict'] in (IGNORE_MARKER, ''):
[9293]790            return 'No verdict in import file.'
[9631]791        # Check if studylevel exists
[9293]792        level_string = str(obj.current_level)
793        if obj.get(level_string) is None:
794            return 'Study level object is missing.'
[9284]795        # Check if student is in state REGISTERED or VALIDATED
[9296]796        if row.get('bypass_validation'):
[9284]797            if obj.student.state not in (VALIDATED, REGISTERED):
798                return 'Student in wrong state.'
799        else:
800            if obj.student.state != VALIDATED:
801                return 'Student in wrong state.'
[7951]802        return None
803
[9706]804    def updateEntry(self, obj, row, site, filename):
[7951]805        """Update obj to the values given in row.
806        """
[8221]807        # Don't set current_session, current_level
808        vals_to_set = dict((key, val) for key, val in row.items()
809                           if key not in ('current_session','current_level'))
[9706]810        super(StudentVerdictProcessor, self).updateEntry(
811            obj, vals_to_set, site, filename)
[7951]812        parent = self.getParent(row, site)
[9293]813        # Set current_vedict in corresponding studylevel
814        level_string = str(obj.current_level)
815        obj[level_string].level_verdict = row['current_verdict']
[9296]816        # Fire transition and set studylevel attributes
817        # depending on student's state
[9284]818        if obj.__parent__.state == REGISTERED:
[9296]819            validated_by = row.get('validated_by', '')
820            if validated_by in (IGNORE_MARKER, ''):
821                portal_language = getUtility(IKofaUtils).PORTAL_LANGUAGE
822                system = translate(_('System'),'waeup.kofa',
823                                  target_language=portal_language)
824                obj[level_string].validated_by = system
825            else:
826                obj[level_string].validated_by = validated_by
827            obj[level_string].validation_date = datetime.utcnow()
[9284]828            IWorkflowInfo(obj.__parent__).fireTransition('bypass_validation')
829        else:
830            IWorkflowInfo(obj.__parent__).fireTransition('return')
[7951]831        # Update the students_catalog
832        notify(grok.ObjectModifiedEvent(obj.__parent__))
833        return
Note: See TracBrowser for help on using the repository browser.