## $Id: test_browser.py 14255 2016-11-03 08:23:24Z henrik $ ## ## Copyright (C) 2011 Uli Fouquet & Henrik Bettermann ## This program is free software; you can redistribute it and/or modify ## it under the terms of the GNU General Public License as published by ## the Free Software Foundation; either version 2 of the License, or ## (at your option) any later version. ## ## This program is distributed in the hope that it will be useful, ## but WITHOUT ANY WARRANTY; without even the implied warranty of ## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ## GNU General Public License for more details. ## ## You should have received a copy of the GNU General Public License ## along with this program; if not, write to the Free Software ## Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA ## import os import shutil import tempfile from mechanize import LinkNotFoundError from StringIO import StringIO from hurry.workflow.interfaces import IWorkflowState, IWorkflowInfo from zope.component.hooks import setSite, clearSite from zope.component import getUtility, createObject from zope.interface import verify from waeup.kofa.app import University from waeup.kofa.students.tests.test_browser import ( StudentsFullSetup, SAMPLE_IMAGE) from waeup.kofa.students.accommodation import BedTicket from waeup.kofa.testing import FunctionalTestCase from waeup.kofa.interfaces import ( IExtFileStore, IFileStoreNameChooser) from waeup.kofa.browser.tests.test_pdf import samples_dir from waeup.kofa.students.interfaces import IStudentsUtils from waeup.kwarapoly.testing import FunctionalLayer from waeup.kwarapoly.students.utils import ( local_nonlocal, arts_science, we_ft) class StudentProcessorTest(FunctionalTestCase): """Perform some batching tests. """ layer = FunctionalLayer def setUp(self): super(StudentProcessorTest, self).setUp() # Setup a sample site for each test app = University() self.dc_root = tempfile.mkdtemp() app['datacenter'].setStoragePath(self.dc_root) # Prepopulate the ZODB... self.getRootFolder()['app'] = app # we add the site immediately after creation to the # ZODB. Catalogs and other local utilities are not setup # before that step. self.app = self.getRootFolder()['app'] # Set site here. Some of the following setup code might need # to access grok.getSite() and should get our new app then setSite(app) def tearDown(self): super(StudentProcessorTest, self).tearDown() shutil.rmtree(self.workdir) shutil.rmtree(self.dc_root) clearSite() return class StudentUITests(StudentsFullSetup): """Tests for customized student class views and pages """ layer = FunctionalLayer def setUp(self): super(StudentUITests, self).setUp() self.certificate.study_mode = 'hnd_ft' bedticket = BedTicket() bedticket.booking_session = 2004 bedticket.bed_type = u'any bed type' bedticket.bed = self.app['hostels']['hall-1']['hall-1_A_101_A'] bedticket.bed_coordinates = u'My bed coordinates' self.student['accommodation'].addBedTicket(bedticket) self.app['configuration']['2004'].gown_fee = 150.0 self.app['configuration']['2004'].transfer_fee = 90.0 self.app['configuration']['2004'].clearance_fee = 120.0 self.app['configuration']['2004'].booking_fee = 150.0 self.app['configuration']['2004'].maint_fee = 180.0 self.app['configuration']['2004'].certificate_fee = 444.0 def test_manage_payments(self): # Managers can add online payment tickets self.browser.addHeader('Authorization', 'Basic mgr:mgrpw') self.browser.open(self.payments_path) self.browser.getLink("Add current session payment ticket").click() self.browser.getControl(name="form.p_category").value = ['schoolfee'] self.browser.getControl("Create ticket").click() self.assertMatches('...Amount could not be determined...', self.browser.contents) IWorkflowState(self.student).setState('cleared') self.browser.open(self.payments_path + '/addop') self.browser.getControl(name="form.p_category").value = ['schoolfee'] self.browser.getControl("Create ticket").click() self.assertMatches('...Book and pay for accommodation first...', self.browser.contents) # In KwaraPoly only returning students can create school fee payment # without having paid accommodation fee IWorkflowState(self.student).setState('returning') configuration = createObject('waeup.SessionConfiguration') configuration.academic_session = 2005 self.app['configuration'].addSessionConfiguration(configuration) self.browser.getControl(name="form.p_category").value = ['schoolfee'] self.browser.getControl("Create ticket").click() self.assertMatches('...ticket created...', self.browser.contents) ctrl = self.browser.getControl(name='val_id') value = ctrl.options[0] self.browser.getLink(value).click() self.assertMatches('...Amount Authorized...', self.browser.contents) # Managers can open payment slip self.browser.getLink("Download payment slip").click() self.assertEqual(self.browser.headers['Status'], '200 Ok') self.assertEqual(self.browser.headers['Content-Type'], 'application/pdf') # Set ticket paid ticket = self.student['payments'].items()[0][1] ticket.p_state = 'paid' self.browser.open(self.payments_path + '/addop') self.browser.getControl(name="form.p_category").value = ['schoolfee'] self.browser.getControl("Create ticket").click() self.assertMatches('...This type of payment has already been made...', self.browser.contents) # Remove all payments so that we can add a school fee payment again keys = [i for i in self.student['payments'].keys()] for payment in keys: del self.student['payments'][payment] self.browser.open(self.payments_path + '/addop') self.browser.getControl(name="form.p_category").value = ['schoolfee'] self.browser.getControl("Create ticket").click() self.assertMatches('...ticket created...', self.browser.contents) self.browser.open(self.payments_path + '/addop') self.browser.getControl(name="form.p_category").value = ['carryover1'] self.browser.getControl("Create ticket").click() self.assertMatches('...ticket created...', self.browser.contents) self.browser.open(self.payments_path + '/addop') self.browser.getControl(name="form.p_category").value = ['carryover2'] self.browser.getControl("Create ticket").click() self.assertMatches('...ticket created...', self.browser.contents) self.browser.open(self.payments_path + '/addop') self.browser.getControl(name="form.p_category").value = ['carryover3'] self.browser.getControl("Create ticket").click() self.assertMatches('...ticket created...', self.browser.contents) self.browser.open(self.payments_path + '/addop') self.browser.getControl( name="form.p_category").value = ['hostel_maintenance'] self.browser.getControl("Create ticket").click() self.assertMatches('...ticket created...', self.browser.contents) self.browser.open(self.payments_path + '/addop') self.browser.getControl(name="form.p_category").value = ['schoolfee'] self.browser.getControl("Create ticket").click() self.assertMatches('...ticket created...', self.browser.contents) # In state admitted school fee can't be determined IWorkflowState(self.student).setState('admitted') self.browser.open(self.payments_path + '/addop') self.browser.getControl(name="form.p_category").value = ['schoolfee'] self.browser.getControl("Create ticket").click() self.assertMatches('...Amount could not be determined...', self.browser.contents) self.browser.open(self.payments_path + '/addop') self.browser.getControl(name="form.p_category").value = ['certificate'] self.browser.getControl("Create ticket").click() self.assertMatches('...ticket created...', self.browser.contents) def test_student_payments(self): # Login IWorkflowState(self.student).setState('returning') self.browser.open(self.login_path) self.browser.getControl(name="form.login").value = self.student_id self.browser.getControl(name="form.password").value = 'spwd' self.browser.getControl("Login").click() self.browser.open(self.student_path + '/payments') self.assertTrue( 'Add current session payment ticket' in self.browser.contents) self.assertFalse( 'Add previous session payment ticket' in self.browser.contents) return def test_get_returning_data(self): # Student is in level 100, session 2004 with verdict A utils = getUtility(IStudentsUtils) self.assertEqual(utils.getReturningData(self.student),(2005, 200)) self.student['studycourse'].current_verdict = 'C' self.assertEqual(utils.getReturningData(self.student),(2005, 110)) self.student['studycourse'].current_verdict = 'D' self.assertEqual(utils.getReturningData(self.student),(2005, 100)) return def test_set_payment_details(self): utils = getUtility(IStudentsUtils) error, payment = utils.setPaymentDetails('schoolfee',self.student) self.assertEqual(payment, None) self.assertEqual(error, u'Amount could not be determined.') IWorkflowState(self.student).setState('cleared') self.assertEqual(local_nonlocal(self.student), 'non-local') self.assertEqual(arts_science(self.student), 'science') self.assertEqual(we_ft(self.student), 'ft') error, payment = utils.setPaymentDetails('schoolfee',self.student) self.assertEqual(error, 'Book and pay for accommodation first before making' ' school fee payments.') # We add a fake maint. payment ticket to meet the condition maint_payment = createObject('waeup.StudentOnlinePayment') self.student['payments']['any_key'] = maint_payment maint_payment.p_category = 'hostel_maintenance' maint_payment.p_state = 'paid' maint_payment.p_session = 2004 error, payment = utils.setPaymentDetails('schoolfee',self.student) self.assertEqual(payment.p_level, 100) self.assertEqual(payment.p_session, 2004) self.assertEqual(payment.amount_auth, 75500.0) self.assertEqual(payment.p_item, u'CERT1') self.assertEqual(error, None) IWorkflowState(self.student).setState('returning') error, payment = utils.setPaymentDetails('schoolfee',self.student) self.assertEqual('Session configuration object is not available.', error) configuration = createObject('waeup.SessionConfiguration') configuration.academic_session = 2005 self.app['configuration'].addSessionConfiguration(configuration) error, payment = utils.setPaymentDetails('schoolfee',self.student) self.assertEqual(payment.p_level, 200) self.assertEqual(payment.p_session, 2005) self.assertEqual(payment.amount_auth, 35090.0) self.assertEqual(payment.p_item, u'CERT1') self.assertEqual(error, None) configuration.penalty_ug = 5000.0 error, payment = utils.setPaymentDetails('schoolfee',self.student) self.assertEqual(payment.amount_auth, 40090.0) self.assertEqual(error, None) error, payment = utils.setPaymentDetails('clearance',self.student) self.assertEqual(payment.p_level, 100) self.assertEqual(payment.p_session, 2004) self.assertEqual(payment.amount_auth, 120.0) self.assertEqual(payment.p_item, u'CERT1') self.assertEqual(error, None) error, payment = utils.setPaymentDetails('carryover1',self.student) self.assertEqual(payment.p_level, 100) self.assertEqual(payment.p_session, 2004) self.assertEqual(payment.amount_auth, 6000.0) self.assertEqual(payment.p_item, u'One Carry-Over') self.assertEqual(payment.p_category, 'carryover1') self.assertEqual(error, None) error, payment = utils.setPaymentDetails('carryover2',self.student) self.assertEqual(payment.p_level, 100) self.assertEqual(payment.p_session, 2004) self.assertEqual(payment.amount_auth, 10000.0) self.assertEqual(payment.p_item, u'Two Carry-Overs') self.assertEqual(payment.p_category, 'carryover2') self.assertEqual(error, None) error, payment = utils.setPaymentDetails('carryover3',self.student) self.assertEqual(payment.p_level, 100) self.assertEqual(payment.p_session, 2004) self.assertEqual(payment.amount_auth, 15000.0) self.assertEqual(payment.p_item, u'Three Carry-Overs') self.assertEqual(payment.p_category, 'carryover3') self.assertEqual(error, None) error, payment = utils.setPaymentDetails('hostel_maintenance',self.student) self.assertEqual(payment.p_level, 100) self.assertEqual(payment.p_session, 2004) self.assertEqual(payment.amount_auth, 876.0) self.assertEqual(payment.p_item, u'My bed coordinates') self.assertEqual(error, None) self.app['hostels']['hall-1'].maint_fee = 0.0 error, payment = utils.setPaymentDetails('hostel_maintenance',self.student) self.assertEqual(payment.p_level, 100) self.assertEqual(payment.p_session, 2004) self.assertEqual(payment.amount_auth, 180.0) self.assertEqual(payment.p_item, u'My bed coordinates') self.assertEqual(error, None) error, payment = utils.setPaymentDetails('bed_allocation',self.student) self.assertEqual(payment.p_level, 100) self.assertEqual(payment.p_session, 2004) self.assertEqual(payment.amount_auth, 150.0) self.assertEqual(payment.p_item, u'') self.assertEqual(error, None) error, payment = utils.setPaymentDetails('certificate',self.student) self.assertEqual(payment.p_level, 100) self.assertEqual(payment.p_session, 2004) self.assertEqual(payment.amount_auth, 444.0) self.assertEqual(payment.p_item, u'') self.assertEqual(payment.p_category, 'certificate') self.assertEqual(error, None) error, payment = utils.setPaymentDetails('schoolfee',self.student, 2004, 100) self.assertEqual(error, u'Previous session payment not yet implemented.') return def test_student_start_clearance(self): self.browser.open(self.login_path) self.browser.getControl(name="form.login").value = self.student_id self.browser.getControl(name="form.password").value = 'spwd' self.browser.getControl("Login").click() IWorkflowInfo(self.student).fireTransition('admit') self.browser.open(self.student_path + '/change_portrait') image = open(SAMPLE_IMAGE, 'rb') ctrl = self.browser.getControl(name='passportuploadedit') file_ctrl = ctrl.mech_control file_ctrl.add_file(image, filename='my_photo.jpg') self.browser.getControl( name='upload_passportuploadedit').click() self.browser.open(self.student_path + '/start_clearance') # In KwaraPoly the students can just start clearance without entering # an activation code. # Deactivated on 19/10/2016 #self.browser.getControl("Start clearance now").click() #self.assertMatches('...Clearance process has been started...', # self.browser.contents) def test_change_passport(self): self.browser.open(self.login_path) self.browser.getControl(name="form.login").value = self.student_id self.browser.getControl(name="form.password").value = 'spwd' self.browser.getControl("Login").click() IWorkflowState(self.student).setState('cleared') self.browser.open(self.student_path + '/change_portrait') image = open(SAMPLE_IMAGE, 'rb') ctrl = self.browser.getControl(name='passportuploadedit') file_ctrl = ctrl.mech_control file_ctrl.add_file(image, filename='my_photo.jpg') self.browser.getControl( name='upload_passportuploadedit').click() def test_student_accommodation(self): del self.student['accommodation']['2004'] # Login self.browser.open(self.login_path) self.browser.getControl(name="form.login").value = self.student_id self.browser.getControl(name="form.password").value = 'spwd' self.browser.getControl("Login").click() # Students can book accommodation without AC ... self.browser.open(self.acco_path) IWorkflowInfo(self.student).fireTransition('admit') self.browser.getControl("Book accommodation").click() self.assertFalse('Activation Code:' in self.browser.contents) self.browser.getControl("Create bed ticket").click() # Bed is randomly selected but, since there is only # one bed for this student, we know that self.assertMatches('...Hall 1, Block A, Room 101, Bed A...', self.browser.contents) return def test_no_beds_for_iot(self): self.app['faculties']['fac1'].code = 'IOT' del self.student['accommodation']['2004'] # Login self.browser.open(self.login_path) self.browser.getControl(name="form.login").value = self.student_id self.browser.getControl(name="form.password").value = 'spwd' self.browser.getControl("Login").click() self.browser.open(self.acco_path) IWorkflowInfo(self.student).fireTransition('admit') self.browser.getControl("Book accommodation").click() self.browser.getControl("Create bed ticket").click() self.assertTrue('There is no free bed in your category iot_male_fr.' in self.browser.contents) return def test_admission_pdf_slips(self): # Login self.browser.open(self.login_path) self.browser.getControl(name="form.login").value = self.student_id self.browser.getControl(name="form.password").value = 'spwd' self.browser.getControl("Login").click() # admission slip IWorkflowState(self.student).setState('school fee paid') self.browser.open(self.student_path) self.browser.getLink("Download admission letter").click() self.assertEqual(self.browser.headers['Status'], '200 Ok') self.assertEqual(self.browser.headers['Content-Type'], 'application/pdf') path = os.path.join(samples_dir(), 'admission_slip.pdf') open(path, 'wb').write(self.browser.contents) print "Sample PDF admission_slip_slip.pdf written to %s" % path self.browser.open(self.student_path) self.assertRaises( LinkNotFoundError, self.browser.getLink, 'Download admission notification') self.browser.open(self.student_path + '/admission_notification.pdf') self.assertTrue('Not allowed' in self.browser.contents) # admission notification IWorkflowState(self.student).setState('cleared') self.browser.open(self.student_path) self.browser.getLink("Download admission notification").click() self.assertEqual(self.browser.headers['Status'], '200 Ok') self.assertEqual(self.browser.headers['Content-Type'], 'application/pdf') path = os.path.join(samples_dir(), 'admission_notification.pdf') open(path, 'wb').write(self.browser.contents) print "Sample PDF admission_notification.pdf written to %s" % path self.browser.open(self.student_path) self.assertRaises( LinkNotFoundError, self.browser.getLink, 'Download admission letter') self.browser.open(self.student_path + '/admission_slip.pdf') self.assertTrue('Not allowed' in self.browser.contents) return def test_registration_pdf_slips(self): # Student cant login if their password is not set IWorkflowState(self.student).setState('school fee paid') self.browser.open(self.login_path) self.browser.getControl(name="form.login").value = self.student_id self.browser.getControl(name="form.password").value = 'spwd' self.browser.getControl("Login").click() self.browser.getLink("Course of Study").click() self.browser.getLink("Add course list").click() self.assertMatches('...Add current level 100 (Year 1)...', self.browser.contents) self.browser.getControl("Create course list now").click() # A level with one course ticket was created self.browser.getLink("100").click() self.browser.getLink("Download course profile slip").click() self.assertEqual(self.browser.headers['Status'], '200 Ok') self.assertEqual(self.browser.headers['Content-Type'], 'application/pdf') path = os.path.join(samples_dir(), 'course_registration_slip.pdf') open(path, 'wb').write(self.browser.contents) print "Sample PDF course_registration_slip.pdf written to %s" % path self.browser.open(self.student_path) self.browser.getLink("Download registration form").click() self.assertEqual(self.browser.headers['Status'], '200 Ok') self.assertEqual(self.browser.headers['Content-Type'], 'application/pdf') path = os.path.join(samples_dir(), 'registration_slip.pdf') open(path, 'wb').write(self.browser.contents) print "Sample PDF registration_form.pdf written to %s" % path return def test_payment_disabled(self): IWorkflowState(self.student).setState('returning') configuration = createObject('waeup.SessionConfiguration') configuration.academic_session = 2005 self.app['configuration'].addSessionConfiguration(configuration) self.browser.addHeader('Authorization', 'Basic mgr:mgrpw') self.browser.open(self.payments_path) self.browser.getLink("Add current session payment ticket").click() self.browser.getControl(name="form.p_category").value = ['schoolfee'] self.browser.getControl("Create ticket").click() self.assertMatches('...ticket created...', self.browser.contents) self.app['configuration']['2005'].payment_disabled = ['sf_all'] self.browser.getLink("Add current session payment ticket").click() self.browser.getControl(name="form.p_category").value = ['schoolfee'] self.browser.getControl("Create ticket").click() self.assertMatches('...This category of payments has been disabled...', self.browser.contents) self.app['configuration']['2005'].payment_disabled = ['sf_non_pg'] # Non-PG students can't pay ... self.browser.getControl(name="form.p_category").value = ['schoolfee'] self.browser.getControl("Create ticket").click() self.assertMatches('...This category of payments has been disabled...', self.browser.contents) # ... but PG can pay. self.certificate.study_mode = 'pg_ft' self.browser.getControl(name="form.p_category").value = ['schoolfee'] self.browser.getControl("Create ticket").click() self.assertMatches('...ticket created...', self.browser.contents) return def test_student_course_registration(self): # Student cant login if their password is not set IWorkflowState(self.student).setState('school fee paid') self.student['studycourse'].current_level = 200 self.browser.open(self.login_path) self.browser.getControl(name="form.login").value = self.student_id self.browser.getControl(name="form.password").value = 'spwd' self.browser.getControl("Login").click() self.browser.open(self.student_path + '/studycourse/add') # Now students can add the current study level self.assertMatches('...Add current level 200 (Year 2)...', self.browser.contents) self.browser.getControl("Create course list now").click() self.browser.getLink("200").click() self.browser.getLink("Edit course list").click() self.browser.getLink("here").click() self.browser.getControl(name="form.course").value = ['COURSE1'] self.course.credits = 100000 self.browser.getControl("Add course ticket").click() self.assertFalse( 'Total credits exceed ' in self.browser.contents) self.browser.getControl("Register course list").click() self.assertTrue( 'Course list has been registered' in self.browser.contents) self.assertEqual(self.student.state, 'courses registered') return def test_set_matric_number(self): # Login as student self.browser.open(self.login_path) IWorkflowState(self.student).setState('school fee paid') self.browser.open(self.login_path) self.browser.getControl(name="form.login").value = self.student_id self.browser.getControl(name="form.password").value = 'spwd' self.browser.getControl("Login").click() self.assertRaises( LinkNotFoundError, self.browser.getLink, 'Get Matriculation Number') self.student.matric_number = None self.student['studycourse'].entry_mode = 'nd_ft' self.student['studycourse'].entry_session = 2014 self.browser.open(self.student_path) self.assertRaises( LinkNotFoundError, self.browser.getLink, 'Get Matriculation Number') self.student['studycourse'].entry_session = 2015 self.browser.open(self.student_path) self.browser.getLink("Get Matriculation Number").click() self.assertTrue('Matriculation number ND/15/dep1/FT/001 assigned.' in self.browser.contents) self.assertEqual(self.student.matric_number, 'ND/15/dep1/FT/001') self.assertRaises( LinkNotFoundError, self.browser.getLink, 'Get Matriculation Number') # Setting matric number is logged. logfile = os.path.join( self.app['datacenter'].storage, 'logs', 'students.log') logcontent = open(logfile).read() self.assertTrue('W1000000 - waeup.kwarapoly.students.browser.StudentGetMatricNumberView - ' 'W1000000 - ND/15/dep1/FT/001 assigned' in logcontent) return # deactivated on 5/11/15, see ticket 129 def xxx_test_student_fileupload(self): # Students can edit clearance data IWorkflowState(self.student).setState('clearance started') self.browser.open(self.login_path) self.browser.getControl(name="form.login").value = self.student_id self.browser.getControl(name="form.password").value = 'spwd' self.browser.getControl("Login").click() self.browser.getLink("Clearance Data").click() self.browser.getLink("Edit").click() self.browser.getControl("Save and request clearance").click() self.assertMatches('...Required input is missing...', self.browser.contents) self.student.nationality = u'DE' self.browser.open(self.edit_clearance_path) self.browser.getControl("Save and request clearance").click() self.assertTrue( 'Missing: Birth Certificate, Acceptance Letter, LGA Identification, ' 'First Sitting Result, Result Statement, Guarantor/Referee Letter, ' 'Affidavit of Good Conduct' in self.browser.contents) # Students can upload documents ctrl = self.browser.getControl(name='birthcertificateupload') file_obj = open(SAMPLE_IMAGE, 'rb') file_ctrl = ctrl.mech_control file_ctrl.add_file(file_obj, filename='my_birth_certificate.jpg') self.browser.getControl( name='upload_birthcertificateupload').click() self.assertTrue( 'href="http://localhost/app/students/W1000000/birth_certificate"' in self.browser.contents) # The list of missing files is now slightly shorter self.browser.getControl("Save and request clearance").click() self.assertTrue( 'Missing: Acceptance Letter, LGA Identification, ' 'First Sitting Result, Result Statement, Guarantor/Referee Letter, ' 'Affidavit of Good Conduct' in self.browser.contents) return