source: main/waeup.kofa/trunk/src/waeup/kofa/students/interfaces.py @ 9700

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

We need to customize if students are allowed to remove mandatory course tickets. Therefore I added a property attribute removable_by_student.

Allow manager to edit the mandatory attribute.

  • Property svn:keywords set to Id
File size: 19.5 KB
RevLine 
[7191]1## $Id: interfaces.py 9698 2012-11-20 09:08:13Z henrik $
[6621]2##
[7191]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##
[8409]18#from datetime import datetime
[7620]19from zope.component import getUtility
[7256]20from zope.interface import Attribute, Interface
[6621]21from zope import schema
[7620]22from zc.sourcefactory.contextual import BasicContextualSourceFactory
[9217]23from waeup.kofa.browser.interfaces import IStudentNavigationBase
[7811]24from waeup.kofa.interfaces import (
[8409]25    IKofaObject, academic_sessions_vocab, validate_email, ICSVExporter)
[7811]26from waeup.kofa.interfaces import MessageFactory as _
[8176]27from waeup.kofa.schema import TextLineChoice, FormattedDate, PhoneNumber
[7811]28from waeup.kofa.students.vocabularies import (
[7915]29    StudyLevelSource, contextual_reg_num_source, contextual_mat_num_source,
[9420]30    GenderSource, nats_vocab
[7915]31    )
[8160]32from waeup.kofa.payments.interfaces import (
[8174]33    IPaymentsContainer, IOnlinePayment)
[7915]34from waeup.kofa.university.vocabularies import (
[9420]35    CourseSource, StudyModeSource, CertificateSource, SemesterSource)
[6621]36
[7620]37# VerdictSource can't be placed into the vocabularies module because it
38# requires importing IStudentsUtils which then leads to circular imports.
39class VerdictSource(BasicContextualSourceFactory):
40    """A verdicts source delivers all verdicts provided
41    in the portal.
42    """
43    def getValues(self, context):
[7841]44        verdicts_dict = getUtility(IStudentsUtils).VERDICTS_DICT
[8820]45        return sorted(verdicts_dict.keys())
[7620]46
47    def getToken(self, context, value):
48        return value
49
50    def getTitle(self, context, value):
[7841]51        verdicts_dict = getUtility(IStudentsUtils).VERDICTS_DICT
[8820]52        if value != '0':
53            return verdicts_dict[value] + ' (%s)' % value
[7688]54        return verdicts_dict[value]
[7620]55
[7681]56
[7150]57class IStudentsUtils(Interface):
58    """A collection of methods which are subject to customization.
59
60    """
[7841]61    def setReturningData(student):
62        """ This method defines what happens after school fee payment
63        depending on the student's senate verdict.
64
65        In the base configuration current level is always increased
66        by 100 no matter which verdict has been assigned.
67        """
68
[9148]69    def setPaymentDetails(category, student, previous_session=None,
70            previous_level=None,):
[8595]71        """Create Payment object and set the payment data of a student for
72        the payment category specified.
[7150]73
74        """
75
[7186]76    def getAccommodation_details(student):
[7150]77        """Determine the accommodation dates of a student.
78
79        """
80
[7186]81    def selectBed(available_beds):
[7150]82        """Select a bed from a list of available beds.
83
84        In the standard configuration we select the first bed found,
85        but can also randomize the selection if we like.
86        """
87
[7186]88    def renderPDF(view, subject='', filename='slip.pdf',):
[7150]89        """Render pdf slips for various pages.
90
91        """
92
[7819]93class IStudentsContainer(IKofaObject):
[7096]94    """A students container contains university students.
[6692]95
96    """
97    def addStudent(student):
98        """Add an IStudent object and subcontainers.
99
100        """
101
102    def archive(id=None):
103        """Create on-dist archive of students.
104
105        If id is `None`, all students are archived.
106
107        If id contains a single id string, only the respective
108        students are archived.
109
110        If id contains a list of id strings all of the respective
111        students types are saved to disk.
112        """
113
114    def clear(id=None, archive=True):
115        """Remove students of type given by 'id'.
116
117        Optionally archive the students.
118
119        If id is `None`, all students are archived.
120
121        If id contains a single id string, only the respective
122        students are archived.
123
124        If id contains a list of id strings all of the respective
125        student types are saved to disk.
126
127        If `archive` is ``False`` none of the archive-handling is done
128        and respective students are simply removed from the
129        database.
130        """
131
[8408]132    unique_student_id = Attribute("""A unique student id.""")
133
[9217]134class IStudentNavigation(IStudentNavigationBase):
[8735]135    """Interface needed for student navigation, logging, etc.
[7150]136
[6642]137    """
[8736]138    student = Attribute('Student object of context.')
[7150]139
[8735]140    def writeLogMessage(view, message):
141        """Write a view specific log message into students.log.
142
143        """
144
[7819]145class IStudentBase(IKofaObject):
[6631]146    """Representation of student base data.
[7150]147
[6621]148    """
[7062]149    history = Attribute('Object history, a list of messages')
[6637]150    state = Attribute('Returns the registration state of a student')
[6699]151    password = Attribute('Encrypted password of a student')
[9334]152    temp_password = Attribute(
153        'Dictionary with user name, timestamp and encrypted password')
[7203]154    certcode = Attribute('The certificate code of any chosen study course')
155    depcode = Attribute('The department code of any chosen study course')
156    faccode = Attribute('The faculty code of any chosen study course')
[7062]157    current_session = Attribute('The current session of the student')
[9142]158    current_level = Attribute('The current level of the student')
[7641]159    current_mode = Attribute('The current mode of the student')
[9182]160    current_verdict = Attribute('The current verdict of the student')
[7364]161    fullname = Attribute('All name parts separated by hyphens')
162    display_fullname = Attribute('The fullname of an applicant')
[8969]163    is_postgrad = Attribute('True if postgraduate student')
[6637]164
[8983]165    suspended = schema.Bool(
166        title = _(u'Account suspended'),
167        default = False,
[9035]168        required = False,
[8983]169        )
170
[6665]171    student_id = schema.TextLine(
[7723]172        title = _(u'Student Id'),
[6849]173        required = False,
[6665]174        )
175
[7357]176    firstname = schema.TextLine(
[7723]177        title = _(u'First Name'),
[6621]178        required = True,
179        )
180
[7357]181    middlename = schema.TextLine(
[7723]182        title = _(u'Middle Name'),
[7357]183        required = False,
184        )
185
186    lastname = schema.TextLine(
[7723]187        title = _(u'Last Name (Surname)'),
[7357]188        required = True,
189        )
190
[6996]191    sex = schema.Choice(
[7723]192        title = _(u'Sex'),
[6996]193        source = GenderSource(),
194        required = True,
195        )
196
[6788]197    reg_number = TextLineChoice(
[7723]198        title = _(u'Registration Number'),
[6696]199        required = True,
200        readonly = False,
[6788]201        source = contextual_reg_num_source,
[6696]202        )
203
[6788]204    matric_number = TextLineChoice(
[7723]205        title = _(u'Matriculation Number'),
[6750]206        required = False,
207        readonly = False,
[6788]208        source = contextual_mat_num_source,
[6750]209        )
210
[6769]211    adm_code = schema.TextLine(
[7723]212        title = _(u'PWD Activation Code'),
[6769]213        required = False,
[8977]214        readonly = False,
[6769]215        )
216
[7133]217    email = schema.ASCIILine(
[7723]218        title = _(u'Email'),
[7133]219        required = False,
220        constraint=validate_email,
221        )
[8176]222    phone = PhoneNumber(
[7723]223        title = _(u'Phone'),
[7331]224        description = u'',
[7133]225        required = False,
226        )
227
[9334]228    def setTempPassword(user, password):
229        """Set a temporary password (LDAP-compatible) SSHA encoded for
230        officers.
231
232        """
233
234    def getTempPassword():
235        """Check if a temporary password has been set and if it
236        is not expired.
237
238        Return the temporary password if valid,
239        None otherwise. Unset the temporary password if expired.
240        """
241
[9131]242    def transfer(certificate, current_session,
243        current_level, current_verdict):
244        """ Creates a new studycourse and backups the old one.
245
246        """
247
[7993]248class IUGStudentClearance(IKofaObject):
249    """Representation of undergraduate student clearance data.
[7150]250
[6631]251    """
[9543]252    officer_comment = schema.Text(
253        title = _(u"Officer's Comment"),
254        required = False,
[6631]255        )
256
[6695]257    clearance_locked = schema.Bool(
[7723]258        title = _(u'Clearance form locked'),
[6695]259        default = False,
[9035]260        required = False,
[6695]261        )
262
[6769]263    clr_code = schema.TextLine(
[7723]264        title = _(u'CLR Activation Code'),
[6769]265        required = False,
[8977]266        readonly = False,
[6769]267        )
268
[9543]269    date_of_birth = FormattedDate(
270        title = _(u'Date of Birth'),
271        required = True,
272        show_year = True,
273        )
274
[7523]275    nationality = schema.Choice(
[8069]276        vocabulary = nats_vocab,
[7723]277        title = _(u'Nationality'),
[7983]278        required = False,
[7523]279        )
280
[7993]281class IPGStudentClearance(IUGStudentClearance):
282    """Representation of postgraduate student clearance data.
283
284    """
285    employer = schema.TextLine(
286        title = _(u'Employer'),
287        required = False,
288        readonly = False,
289        )
290
[7819]291class IStudentPersonal(IKofaObject):
[6631]292    """Representation of student personal data.
[7150]293
[6631]294    """
[9543]295    personal_updated = schema.Datetime(
296        title = _(u'Updated'),
297        required = False,
298        readonly = False,
299        )
300
[6651]301    perm_address = schema.Text(
[7723]302        title = _(u'Permanent Address'),
[6631]303        required = False,
304        )
305
[8008]306class IStudent(IStudentBase,IUGStudentClearance,IPGStudentClearance,
307    IStudentPersonal):
[6631]308    """Representation of a student.
[7150]309
[6631]310    """
311
[9563]312class IStudentPersonalEdit(IStudentPersonal):
313    """Interface for editing personal data by students.
314
315    Here we can repeat the fields from IStudentPersonal and set the
316    `required` if necessary.
317    """
318
319    perm_address = schema.Text(
320        title = _(u'Permanent Address'),
321        required = True,
322        )
323
[6849]324class IStudentUpdateByRegNo(IStudent):
325    """Representation of a student. Skip regular reg_number validation.
[7150]326
[6849]327    """
328    reg_number = schema.TextLine(
[7723]329        title = _(u'Registration Number'),
[6849]330        required = False,
331        )
332
333class IStudentUpdateByMatricNo(IStudent):
334    """Representation of a student. Skip regular matric_number validation.
[7150]335
[6849]336    """
337    matric_number = schema.TextLine(
[7723]338        title = _(u'Matriculation Number'),
[6849]339        required = False,
340        )
341
[8779]342class IStudentRequestPW(IStudent):
343    """Representation of an student for first-time password request.
344
345    This interface is used when students use the requestpw page to
346    login for the the first time.
347    """
[8854]348    number = schema.TextLine(
349        title = _(u'Registr. or Matric. Number'),
[8779]350        required = True,
351        )
352
353    firstname = schema.TextLine(
354        title = _(u'First Name'),
355        required = True,
356        )
357
358    email = schema.ASCIILine(
359        title = _(u'Email Address'),
360        required = True,
361        constraint=validate_email,
362        )
363
[7819]364class IStudentStudyCourse(IKofaObject):
[6633]365    """A container for student study levels.
366
367    """
[6648]368    certificate = schema.Choice(
[7723]369        title = _(u'Certificate'),
[6648]370        source = CertificateSource(),
[7209]371        required = False,
[6633]372        )
[6635]373
[6996]374    entry_mode = schema.Choice(
[7723]375        title = _(u'Entry Mode'),
[7681]376        source = StudyModeSource(),
[6996]377        required = True,
378        readonly = False,
379        )
380
381    entry_session = schema.Choice(
[7723]382        title = _(u'Entry Session'),
[6996]383        source = academic_sessions_vocab,
[7425]384        #default = datetime.now().year,
[6996]385        required = True,
386        readonly = False,
387        )
388
[6724]389    current_session = schema.Choice(
[7723]390        title = _(u'Current Session'),
[6744]391        source = academic_sessions_vocab,
[6724]392        required = True,
[6996]393        readonly = False,
[6724]394        )
395
396    current_level = schema.Choice(
[7723]397        title = _(u'Current Level'),
[6725]398        source = StudyLevelSource(),
399        required = False,
[6996]400        readonly = False,
[6724]401        )
402
403    current_verdict = schema.Choice(
[7723]404        title = _(u'Current Verdict'),
[7619]405        source = VerdictSource(),
[8820]406        default = '0',
[6725]407        required = False,
[6724]408        )
409
410    previous_verdict = schema.Choice(
[7723]411        title = _(u'Previous Verdict'),
[7619]412        source = VerdictSource(),
[8820]413        default = '0',
[6725]414        required = False,
[6724]415        )
416
[9138]417class IStudentStudyCourseTransfer(IStudentStudyCourse):
418    """An student transfers.
419
420    """
421
422    certificate = schema.Choice(
423        title = _(u'Certificate'),
424        source = CertificateSource(),
425        required = True,
426        )
427
428    current_level = schema.Choice(
429        title = _(u'Current Level'),
430        source = StudyLevelSource(),
431        required = True,
432        readonly = False,
433        )
434
435
436IStudentStudyCourseTransfer['certificate'].order = IStudentStudyCourse[
437    'certificate'].order
438IStudentStudyCourseTransfer['current_level'].order = IStudentStudyCourse[
439    'current_level'].order
440
[7951]441class IStudentVerdictUpdate(IKofaObject):
442    """A interface for verdict imports.
443
444    """
445
446    current_verdict = schema.Choice(
447        title = _(u'Current Verdict'),
448        source = VerdictSource(),
449        required = True,
450        )
451
452    current_session = schema.Choice(
453        title = _(u'Current Session'),
454        source = academic_sessions_vocab,
455        required = True,
456        )
457
458    current_level = schema.Choice(
459        title = _(u'Current Level'),
460        source = StudyLevelSource(),
461        required = True,
462        )
463
[9296]464    bypass_validation = schema.Bool(
465        title = _(u'Bypass validation'),
466        required = False,
467        )
468
469    validated_by = schema.TextLine(
470        title = _(u'Validated by'),
471        required = False,
472        )
473
[7819]474class IStudentStudyLevel(IKofaObject):
[6774]475    """A container for course tickets.
476
477    """
478    level = Attribute('The level code')
[9235]479    number_of_tickets = Attribute('Number of tickets contained in this level')
[9253]480    certcode = Attribute('The certificate code of the study course')
[9257]481    is_current_level = Attribute('Is this level the current level of the student?')
[6774]482
[6793]483    level_session = schema.Choice(
[7723]484        title = _(u'Session'),
[6793]485        source = academic_sessions_vocab,
[9437]486        required = True,
[6793]487        )
[6781]488
[6793]489    level_verdict = schema.Choice(
[7723]490        title = _(u'Verdict'),
[7619]491        source = VerdictSource(),
[8820]492        default = '0',
[6793]493        required = False,
494        )
495
[9161]496    validated_by = schema.TextLine(
497        title = _(u'Validated by'),
498        default = None,
499        required = False,
500        )
501
502    validation_date = schema.Datetime(
503        title = _(u'Validation Date'),
504        required = False,
505        readonly = False,
506        )
507
[9690]508    total_credits = schema.Int(
509        title = _(u'Total Credits'),
510        required = False,
511        readonly = True,
512        )
513
514    gpa = schema.Int(
515        title = _(u'GPA'),
516        required = False,
517        readonly = True,
518        )
519
[8920]520    def addCourseTicket(ticket, course):
[8182]521        """Add a course ticket object.
522        """
523
[9501]524    def addCertCourseTickets(cert):
525        """Collect all certificate courses and create course
526        tickets automatically.
527        """
528
[7819]529class ICourseTicket(IKofaObject):
[9420]530    """An interface for course tickets.
[6781]531
532    """
[6783]533    code = Attribute('code of the original course')
[9253]534    certcode = Attribute('certificate code of the study course')
[9684]535    grade = Attribute('grade calculated from score')
536    weight = Attribute('weight calculated from score')
[9698]537    removable_by_student = Attribute('Is student allowed to remove the ticket?')
[6781]538
[9420]539    title = schema.TextLine(
540        title = _(u'Title'),
541        required = False,
542        )
543
544    fcode = schema.TextLine(
545        title = _(u'Faculty Code'),
546        required = False,
547        )
548
549    dcode = schema.TextLine(
550        title = _(u'Department Code'),
551        required = False,
552        )
553
554    semester = schema.Choice(
555        title = _(u'Semester/Term'),
556        source = SemesterSource(),
557        required = False,
558        )
559
560    passmark = schema.Int(
561        title = _(u'Passmark'),
562        required = False,
563        )
564
565    credits = schema.Int(
566        title = _(u'Credits'),
567        required = False,
568        )
569
[7665]570    mandatory = schema.Bool(
[9320]571        title = _(u'Required'),
[6795]572        default = False,
573        required = False,
574        )
575
[6781]576    score = schema.Int(
[7723]577        title = _(u'Score'),
[9684]578        default = None,
[6781]579        required = False,
580        )
581
[7661]582    carry_over = schema.Bool(
[7723]583        title = _(u'Carry-over Course'),
[7661]584        default = False,
585        required = False,
586        )
587
[9420]588    automatic = schema.Bool(
589        title = _(u'Automatical Creation'),
590        default = False,
[9316]591        required = False,
592        )
593
[7633]594    def getLevel():
595        """Returns the id of the level the ticket has been added to.
596        """
597
598    def getLevelSession():
599        """Returns the session of the level the ticket has been added to.
600        """
601
[9420]602class ICourseTicketAdd(IKofaObject):
[7150]603    """An interface for adding course tickets.
[6795]604
605    """
606    course = schema.Choice(
[7723]607        title = _(u'Course'),
[6795]608        source = CourseSource(),
609        readonly = False,
610        )
611
[9420]612class ICourseTicketImport(ICourseTicket):
613    """An interface for importing course results and nothing more.
614
615    """
616    score = schema.Int(
617        title = _(u'Score'),
618        required = False,
619        readonly = False,
620        )
621
622    level_session = schema.Choice(
623        title = _(u'Level Session'),
624        source = academic_sessions_vocab,
625        required = False,
626        readonly = False,
627        )
628
[7819]629class IStudentAccommodation(IKofaObject):
[6635]630    """A container for student accommodation objects.
631
632    """
633
[9423]634    def addBedTicket(bedticket):
635        """Add a bed ticket object.
636        """
637
638
[7819]639class IBedTicket(IKofaObject):
[6989]640    """A ticket for accommodation booking.
641
642    """
[6996]643    bed = Attribute('The bed object.')
644
645    bed_coordinates = schema.TextLine(
[9424]646        title = _(u'Allocated Bed'),
[9423]647        required = True,
[7014]648        readonly = False,
[6992]649        )
650
[6996]651    bed_type = schema.TextLine(
[9424]652        title = _(u'Requested Bed Type'),
[9423]653        required = True,
[7014]654        readonly = False,
[6996]655        )
656
[6992]657    booking_session = schema.Choice(
[7723]658        title = _(u'Session'),
[6992]659        source = academic_sessions_vocab,
660        required = True,
[7014]661        readonly = True,
[6992]662        )
663
[8170]664    booking_date = schema.Datetime(
[7723]665        title = _(u'Booking Date'),
[6992]666        required = False,
[7014]667        readonly = True,
[6992]668        )
669
670    booking_code = schema.TextLine(
[7723]671        title = _(u'Booking Activation Code'),
[6992]672        required = False,
[7014]673        readonly = True,
[6992]674        )
675
[6994]676    def getSessionString():
[7633]677        """Returns the title of academic_sessions_vocab term.
[7150]678
[6994]679        """
[6992]680
[6860]681class IStudentPaymentsContainer(IPaymentsContainer):
[6635]682    """A container for student payment objects.
683
684    """
685
[6877]686class IStudentOnlinePayment(IOnlinePayment):
687    """A student payment via payment gateways.
688
689    """
690
[9148]691    p_current = schema.Bool(
692        title = _(u'Current Session Payment'),
693        default = True,
694        required = False,
695        )
696
[8268]697    p_level = schema.Int(
698        title = _(u'Payment Level'),
699        required = False,
700        )
[6877]701
[8422]702    def doAfterStudentPayment():
703        """Process student after payment was made.
704
705        """
706
[8453]707    def doAfterStudentPaymentApproval():
708        """Process student after payment was approved.
709
710        """
711
[8420]712    def approveStudentPayment():
[8422]713        """Approve payment and process student.
[8420]714
715        """
716
[8268]717IStudentOnlinePayment['p_level'].order = IStudentOnlinePayment[
718    'p_session'].order
[8408]719
[9148]720class IStudentPreviousPayment(IOnlinePayment):
721    """An interface for adding previous session payments.
722
723    """
724
725    p_session = schema.Choice(
726        title = _(u'Payment Session'),
727        source = academic_sessions_vocab,
728        required = True,
729        )
730
731    p_level = schema.Choice(
732        title = _(u'Payment Level'),
733        source = StudyLevelSource(),
734        required = True,
735        )
736
[8408]737class ICSVStudentExporter(ICSVExporter):
738    """A regular ICSVExporter that additionally supports exporting
739      data from a given student object.
740    """
741
742    def export_student(student, filepath=None):
743        """Export data for a given student.
744        """
Note: See TracBrowser for help on using the repository browser.