:mod:`waeup.sirp.accesscodes.browser` -- UI components for access-codes *********************************************************************** .. module:: waeup.sirp.accesscodes.browser Here we visit the access-code related parts of a WAeUP SIRP site using a virtual browser. :NOTTest-Layer: functional Preliminaries ============= Before we can do anything, we have to create a university site: >>> from zope.testbrowser.testing import Browser >>> browser = Browser() >>> browser.addHeader('Authorization', 'Basic mgr:mgrpw') >>> browser.handleErrors = False >>> root = getRootFolder() >>> from waeup.sirp.app import University >>> u = University() >>> root['myuniversity'] = u We set a new datacenter storage path: >>> import os >>> browser.open('http://localhost/myuniversity') >>> browser.getLink('Data Center').click() >>> browser.getLink('Edit settings').click() >>> pathsetting = browser.getControl(name='newpath') >>> cwd = os.getcwd() >>> uploadpath = os.path.join(cwd, 'ac_testfiles') >>> os.mkdir(uploadpath) >>> pathsetting.value = uploadpath >>> browser.getControl(name='save').click() We remove any existing 'accesscodes' dir from datacenter dir: >>> import shutil >>> if os.path.exists(os.path.join(uploadpath, 'accesscodes')): ... shutil.rmtree(os.path.join(uploadpath, 'accesscodes')) Access-code management screen ============================= For users that have the right to manage access-code related stuff, the home page of a `University` instance provides a link to the access-code management screen. In the beginning, there are naturally no batches available: >>> browser.open('http://localhost/myuniversity') >>> browser.getLink('Access Codes').click() >>> print browser.contents Access Code Batches ... ... The following batches are available: ...No batches yet... ... Adding batches ============== We can add a batch of access-codes using a button displayed in the action bar: >>> browser.getLink('Add Scratch Card Batch').click() The add screen shows a form where we have to enter a prefix, the number of access codes to be generated and the costs for each card. >>> browser.getControl(name='form.prefix').value = 'APP' >>> browser.getControl(name='form.entry_num').value = '5' >>> browser.getControl(name='form.cost').value = '12.12' If we click 'cancel' afterwards, the whole process will be cancelled and we'll be redirected to the management screen: >>> browser.getControl('Cancel').click() >>> browser.url 'http://localhost/myuniversity/accesscodes' >>> 'Batch creation cancelled' in browser.contents True Now let's try again and this time we finish the procedure by clicking 'Create batch' in the form: >>> browser.getLink('Add Scratch Card Batch').click() >>> browser.getControl(name='form.prefix').value = 'APP' >>> browser.getControl(name='form.entry_num').value = '5' >>> browser.getControl(name='form.cost').value = '12.12' >>> browser.getControl('Create batch').click() We're also redirected to the management screen, with a notice about the freshly created batch: >>> print browser.contents Access Code Batches ... ... The following batches are available: ...APP ...- ...1 ... ...5 .../ ...0 ... ...12.12... ...zope.mgr... ... which means: there exists a batch named ``APP-1`` with 5 entries of which 0 have been devalidated, each one costs ``12.12`` and the batch was created by ``zope.mgr``. We create a second batch to see whether searching and related stuff works: >>> browser.getLink('Add Scratch Card Batch').click() >>> browser.getControl(name='form.prefix').value = 'APP' >>> browser.getControl(name='form.entry_num').value = '5' >>> browser.getControl(name='form.cost').value = '10.12' >>> browser.getControl('Create batch').click() And a third one that can be deleted afterwards: >>> browser.getLink('Add Scratch Card Batch').click() >>> browser.getControl(name='form.prefix').value = 'BLA' >>> browser.getControl(name='form.entry_num').value = '3' >>> browser.getControl(name='form.cost').value = '19.12' >>> browser.getControl('Create batch').click() We also invalidate one entry: >>> batch = getRootFolder()['myuniversity']['accesscodes']['BLA-1'] >>> ac = list(batch.entries())[0] >>> batch.invalidate(ac.representation) Creating Archive Files ====================== Once a batch is created, we can archive it. To do so we have to tick the respective checkbox and click on 'Archive': >>> ctrl = browser.getControl(name='batches') >>> ctrl.getControl(value='APP-2').selected = True >>> browser.getControl(name='archive').click() >>> print browser.contents Archived APP-2 (APP-2_archive-...-zope.mgr.csv) ... If we do not select a batch and try to archive or delete, the system will complain: >>> browser.getControl(name='archive').click() >>> print browser.contents No batch selected. ... Deleting Batches ================ We can delete batches. They are automatically archived when doing so: >>> ctrl = browser.getControl(name='batches') >>> ctrl.getControl(value='BLA-1').selected = True >>> browser.getControl('Archive and Delete').click() >>> print browser.contents >> browser.getLink('Reimport Scratch Card Batch').click() >>> print browser.contents >> browser.getControl('Cancel').click() We copy the ``BLA-1`` batch logfile over to the ``imports`` directory: >>> ac_storage = os.path.join(uploadpath, 'accesscodes') >>> logfile2 = os.path.join(ac_storage, ... sorted(os.listdir(ac_storage))[-3]) >>> filename = os.path.basename(logfile2) >>> filename 'BLA-1-...-zope.mgr.csv' >>> import shutil >>> import_path = os.path.join(ac_storage, 'imports') >>> shutil.copy(logfile2, import_path) Now the file will be presented as import source: >>> browser.getLink('Reimport Scratch Card Batch').click() >>> filename in browser.contents True If we do not tick a filename to import and click 'Reimport', we will be warned: >>> browser.getControl('Reimport').click() >>> print browser.contents >> browser.getLink('Reimport Scratch Card Batch').click() >>> ctrl = browser.getControl(name='filenames') >>> ctrl.getControl(value=filename).selected = True >>> browser.getControl('Reimport').click() The batch does exist now again: >>> print browser.contents >> browser.getLink('Reimport Scratch Card Batch').click() >>> ctrl = browser.getControl(name='filenames') >>> ctrl.getControl(value=filename).selected = True >>> browser.getControl('Reimport').click() >>> print browser.contents >> ac_storage = os.path.join(uploadpath, 'accesscodes') Log files for access-code batches --------------------------------- Whenever a batch is created, there is also a log file with all entries created: >>> sorted(os.listdir(ac_storage)) ['APP-1-...-zope.mgr.csv', 'APP-2-...-zope.mgr.csv', ...] Each logfile name contains the prefix, batch number, date of creation and userid of creator. >>> logfile1 = os.path.join(ac_storage, ... sorted(os.listdir(ac_storage))[0]) >>> print open(logfile1, 'rb').read() "serial","ac","cost" "APP","1","12.12" "0","APP-1-<10-DIGITS>" "1","APP-1-<10-DIGITS>" "2","APP-1-<10-DIGITS>" "3","APP-1-<10-DIGITS>" "4","APP-1-<10-DIGITS>" Archive files ------------- We created an archive file above. An archive file name consists of the batch prefix, batch number, the string ``_archive``, creation datetime of the archive file and userid of batch creator: >>> sorted(os.listdir(ac_storage)) [..., 'BLA-1_archive-...-zope.mgr.csv', 'imports'] Archive files eventually also contain infos about invalidation dates and have a slightly different format therefore. >>> archive_file = os.path.join(ac_storage, ... sorted(os.listdir(ac_storage))[-2]) >>> print open(archive_file, 'rb').read() "prefix","serial","ac","student","date" "BLA","19.12","1","3" "BLA","0","BLA-1-<10-DIGITS>","","..." "BLA","1","BLA-1-<10-DIGITS>","","" "BLA","2","BLA-1-<10-DIGITS>","","" Clean up: >>> shutil.rmtree(uploadpath)