source: WAeUP_SRP/base/WAeUPTool.py @ 2316

Last change on this file since 2316 was 2315, checked in by Henrik Bettermann, 17 years ago

import.py fixed
logging in importData improved

  • Property svn:keywords set to Id
File size: 62.3 KB
Line 
1#-*- mode: python; mode: fold -*-
2# (C) Copyright 2005 The WAeUP group  <http://www.waeup.org>
3# Author: Joachim Schmitz (js@aixtraware.de)
4#
5# This program is free software; you can redistribute it and/or modify
6# it under the terms of the GNU General Public License version 2 as published
7# by the Free Software Foundation.
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
17# 02111-1307, USA.
18#
19# $Id: WAeUPTool.py 2315 2007-10-05 06:19:47Z henrik $
20"""The WAeUP Tool Box.
21"""
22
23from AccessControl import ClassSecurityInfo
24from Acquisition import aq_inner
25from Acquisition import aq_parent
26from Globals import DTMLFile
27from Globals import InitializeClass
28from OFS.SimpleItem import SimpleItem
29from zExceptions import BadRequest
30
31from Products.CMFCore.utils import getToolByName
32from Products.CPSSchemas.DataStructure import DataStructure
33from Products.CPSSchemas.DataModel import DataModel
34from Products.CPSSchemas.StorageAdapter import MappingStorageAdapter
35from Products.CMFCore.ActionProviderBase import ActionProviderBase
36from Products.CMFCore.permissions import View
37from Products.ZCatalog.ZCatalog import ZCatalog
38from Products.CMFCore.permissions import ModifyPortalContent
39from Products.CMFCore.permissions import ManagePortal
40from Products.CMFCore.utils import UniqueObject
41from Products.CMFCore.URLTool import URLTool
42from Products.CMFCore.utils import getToolByName
43from Students import makeCertificateCode
44from Globals import package_home,INSTANCE_HOME
45p_home = package_home(globals())
46i_home = INSTANCE_HOME
47import DateTime,time
48import logging
49import transaction
50import csv,re,os,sys
51from Products.AdvancedQuery import Eq, Between, Le,In
52
53def getObject(object,name):
54    if object.hasObject(name):
55        return getattr(object,name)
56    return None
57
58class WAeUPTool(UniqueObject, SimpleItem, ActionProviderBase):
59    """WAeUP tool"""
60
61    id = 'waeup_tool'
62    meta_type = 'WAeUP Tool'
63    _actions = ()
64    security = ClassSecurityInfo()
65    security.declareObjectProtected(View)
66    manage_options = ( ActionProviderBase.manage_options
67                     + SimpleItem.manage_options
68                     )
69
70    def rwrite(self,s): ###(
71        response = self.REQUEST.RESPONSE
72        response.setHeader('Content-type','text/html; charset=ISO-8859-15')
73        response.write("%s<br />\r\n" % s)
74    ###)
75
76    def sleep(self,secs): ###(
77        "sleep"
78        import time
79        time.sleep(secs)
80        return
81
82###)
83
84    security.declareProtected(ModifyPortalContent,'openLog') ###(
85    def openLog(self,name):
86        """open a log file"""
87        version = 1
88        path = "%s/log/%s_%d.log" % (i_home,name,version)
89        while os.path.exists(path):
90            version += 1
91            path = "%s/log/%s_%d.log" % (i_home,name,version)
92        log = open(path,"w")
93        return log
94    ###)
95
96    security.declareProtected(ModifyPortalContent,'bypassQueueCatalog') ###(
97    def bypassQueueCatalog(self,enable=True):
98        """bypass the QueueCatalog by setting all indexes to process imediate,
99        if enable is True (default) the old settings are restored
100        """
101
102    ###)
103
104    security.declareProtected(ModifyPortalContent,'measureOaT') ###(
105    def measureOaT(self,method="a",probe="1000",nr_pts="1"):
106        """measure Object access Time"""
107        import random
108        if hasattr(self,'portal_catalog_real'):
109            aq_portal = self.portal_catalog_real.evalAdvancedQuery
110        else:
111            aq_portal = self.portal_catalog.evalAdvancedQuery
112        nr_pts = int(nr_pts)
113        probe = int(probe)
114        intervall = probe/10
115        objects = ("application","clearance","personal")
116        portal_types = ("StudentApplication","StudentClearance","StudentPersonal")
117        #i = random.randrange(num_objects)
118        count = 0
119        found = 0
120        not_found = 0
121        t_found = 0
122        t_not_found = 0
123        time_found = time_not_found = 0.0
124        t_time_found = t_time_not_found = 0.0
125        accessed = []
126        t_min = 1000
127        t_max = 0
128        #import pdb;pdb.set_trace()
129        students = self.portal_catalog(portal_type="Student")
130        num_students = len(students)
131        if method == "d":
132            query = Eq('path','/uniben/campus/students') & In('portal_type',portal_types[:nr_pts])
133            res = aq_portal(query)
134            brains = {}
135            for r in res:
136                sid = r.relative_path.split('/')[-2]
137                if brains.has_key(sid):
138                    brains[sid][r.portal_type] = r
139                else:
140                    brains[sid] = {r.portal_type : r}
141            brains_list = brains.keys()
142            num_objects = len(brains_list)
143        else:
144            num_objects = num_students
145        print "="*40
146        print "method: %s probes: %d nr_pts: %d num_objects: %d" % (method,
147                                                                        probe,
148                                                                        nr_pts,
149                                                                        num_objects)
150        print "nr found/not time found/not min/max"
151        elapse = time.time()
152        i_elapse = time.time()
153        c_elapse = time.clock()
154        for c in range(1,probe + 1):
155            i = random.randrange(num_objects)
156            if method in ('a','b','c'):
157                student_brain = students[i]
158            elif method == "d":
159                #import pdb;pdb.set_trace()
160                student_brain = brains[brains_list[i]]
161            if method == "c":
162                query = Eq('path',student_brain.getPath()) & In('portal_type',portal_types[:nr_pts])
163                res = aq_portal(query)
164                this_portal_types = [r.portal_type for r in res]
165            for i in range(nr_pts):
166                oid = objects[i]
167                if method == "a":
168                    try:
169                        student_path = student_brain.getPath()
170                        path = "%s/%s" % (student_path,oid)
171                        doc = self.unrestrictedTraverse(path).getContent()
172                        found += 1
173                        i_time = time.time() - i_elapse
174                        time_found += i_time
175                    except:
176                        not_found += 1
177                        i_time = time.time() - i_elapse
178                        time_not_found += i_time
179                        pass
180                elif method == "b":
181                    try:
182                        student_object = student_brain.getObject()
183                        doc = getattr(student_object,oid).getContent()
184                        found += 1
185                        i_time = time.time() - i_elapse
186                        time_found += i_time
187                    except:
188                        i_time = time.time() - i_elapse
189                        time_not_found += i_time
190                        not_found += 1
191                        pass
192                elif method == "c":
193                    if portal_types[i] in this_portal_types:
194                        found += 1
195                        doc = res[this_portal_types.index(portal_types[i])].getObject().getContent()
196                        i_time = time.time() - i_elapse
197                        time_found += i_time
198                    else:
199                        not_found += 1
200                        i_time = time.time() - i_elapse
201                        time_not_found += i_time
202                elif method == "d":
203                    if student_brain.has_key(portal_types[i]):
204                        found += 1
205                        doc = student_brain[portal_types[i]].getObject().getContent()
206                        i_time = time.time() - i_elapse
207                        time_found += i_time
208                    else:
209                        not_found += 1
210                        i_time = time.time() - i_elapse
211                        time_not_found += i_time
212                i_elapse = time.time()
213            if c and (c % intervall == 0):
214                #i_time = time.time() - i_elapse
215                t_per = 0.0
216                if found:
217                    t_per = time_found/found
218                if t_per > t_max:
219                    t_max = t_per
220                if t_per > 0.0 and t_per < t_min:
221                    t_min = t_per
222                itf = 0.0
223                if found:
224                    itf = time_found/found
225                itnf = 0.0
226                if not_found :
227                    itnf = time_not_found / not_found
228                interval_time = time_found + time_not_found
229                s = "%(c)d: %(found)d/%(not_found)d " % vars()
230                s += "%(interval_time)6.2f %(itf)6.4f/%(itnf)6.4f " % vars()
231                s += "%(t_min)6.4f/%(t_max)6.4f" %  vars()
232                print s
233                t_found += found
234                t_not_found += not_found
235                t_time_found += time_found
236                t_time_not_found += time_not_found
237                time_found = time_not_found = 0.0
238                found = not_found = 0
239        # t_found += found
240        # t_not_found += not_found
241        elapse = time.time() - elapse
242        itf = 0.0
243        if t_found:
244            itf = t_time_found/t_found
245        itnf = 0.0
246        if t_not_found:
247            itnf = t_time_not_found / t_not_found
248        #c_elapse = time.clock() - c_elapse
249        s = "%(probe)d: %(t_found)d/%(t_not_found)d " % vars()
250        s += "%(elapse)6.2f %(itf)6.4f/%(itnf)6.4f " % vars()
251        s += "%(t_min)6.4f/%(t_max)6.4f" %  vars()
252        print "-"*40
253        print s
254        rel_found = float(t_found)/probe
255        rel_not_found = float(t_not_found)/probe
256        estimated_total_time = num_objects*(rel_found*itf + rel_not_found*itnf)
257        print estimated_total_time
258    ###)
259
260    security.declareProtected(ModifyPortalContent,'writeLog') ###(
261    def writeLog(self,logfile,s):
262        """write to the log file"""
263        logfile.write(s)
264
265###)
266
267    def generateStudentId(self,letter): ###(
268        import random
269        r = random
270        ##if letter not in ('ABCDEFGIHKLMNOPQRSTUVWXY'):
271        if letter == '?':
272            letter= r.choice('ABCDEFGHKLMNPQRSTUVWXY')
273        sid = "%c%d" % (letter,r.randint(99999,1000000))
274        students = self.portal_url.getPortalObject().campus.students
275##        while hasattr(students, sid):
276##            sid = "%c%d" % (letter,r.randint(99999,1000000))
277        while self.students_catalog(id = sid):
278            sid = "%c%d" % (letter,r.randint(99999,1000000))
279        return sid
280    ###)
281
282    def generatePassword(self,s=None): ###(
283        import random
284        r = random
285        ##if letter not in ('ABCDEFGIHKLMNOPQRSTUVWXY'):
286        if s is None:
287            s = 'abcdefghklmnpqrstuvwxy23456789'
288        pw = ''
289        while len(pw) < 6:
290            pw += r.choice(s)
291        return pw
292    ###)
293
294    security.declareProtected(ManagePortal, 'removeDeletedDocIds') ###(
295    def removeDeletedDocIds(self, max=1000):
296        """
297        remove deleted docids from repository commit after max
298        """
299        logger = logging.getLogger('WAeUPTool.removeDeletedDocIds')
300        repository = getToolByName(self, 'portal_repository')
301        pxtool = getToolByName(self, 'portal_proxies')
302        pxtool_infos = pxtool.getRevisionsUsed()
303
304        nb_revs = 0
305        docids_d = {} # all docids
306        unused_docids_d = {} # all docids that are unused
307        ids_unused_revs_docids = [] # ids for revs of unused docids
308        ids_unused_revs = [] # ids for unused revs
309        total = 0
310        idlist = repository.objectIds()
311        for id in idlist:
312            docid, rev = repository._splitId(id)
313            if docid is None:
314                logger.info("invalid doc_id %s" % docid)
315                continue
316            nb_revs += 1
317            docids_d[docid] = None
318            if not pxtool_infos.has_key(docid):
319                unused_docids_d[docid] = None
320                ids_unused_revs_docids.append(id)
321                ids_unused_revs.append(id)
322            elif not pxtool_infos[docid].has_key(rev):
323                ids_unused_revs.append(id)
324            if len(ids_unused_revs) >= max:
325                repository.manage_delObjects(ids_unused_revs)
326                #import pdb;pdb.set_trace()
327                transaction.commit()
328                total += max
329                logger.info('removed %d total %d unused docids ' % (max,total))
330        anz = len(ids_unused_revs)
331        if anz > 0:
332            repository.manage_delObjects(ids_unused_revs)
333            transaction.commit()
334            total += anz
335            logger.info('finished removing %d unused docids ' % (total))
336
337
338###)
339
340    security.declareProtected(View,'getCredential') ###(
341    def getCredential(self,student_id):
342        "return a student password"
343        student_entry = getattr(self.portal_directories.students,student_id,None)
344        if student_entry is None:
345            return None
346        return getattr(student_entry,"password","not set")
347    ###)
348
349    security.declarePublic('checkPassword') ###(
350    def checkPassword(self,student_id,password):
351        "return a student password"
352        student_entry = getattr(self.portal_directories.students,student_id,None)
353        if student_entry is None:
354            return False
355        return getattr(student_entry,"password","not set") == password
356    ###)
357
358    security.declarePublic('editPassword') ###(
359    def editPassword(self,student_id,password):
360        "edit a student password"
361        student_entry = getattr(self.portal_directories.students,student_id,None)
362        if student_entry is None:
363            return
364        setattr(student_entry,'password',password)
365    ###)
366
367    security.declareProtected(View,'doCommit') ###(
368    def doCommit(self,logger=None):
369        "commit some transactions"
370        transaction.commit()
371    ###)
372
373    security.declarePublic('loadStudentFoto') ###(
374    def loadStudentFoto(self,student,filename,folder):
375        "return a student passport picture"
376        app = student.application
377        app_doc = app.getContent()
378        #clear = student.clearance
379        #clear_doc = clear.getContent()
380        #matric_no = clear_doc.matric_no.upper()
381        picture1 ="%s/import/%s/%s.jpg" % (i_home,folder,filename)
382        picture2 ="%s/import/%s/%s.JPG" % (i_home,folder,filename)
383        #import pdb;pdb.set_trace()
384        if os.path.exists(picture1):
385            file = open(picture1)
386        elif os.path.exists(picture2):
387            file = open(picture2)
388        else:
389            return "passport picture not found %s" % picture1
390        reopened = False
391        if self.portal_workflow.getInfoFor(app,'review_state',None) !='opened':
392            self.portal_workflow.doActionFor(app,'open')
393            reopened = True
394        outfile = file.read()
395        app_doc.manage_addFile('passport',
396                               file=outfile,
397                               title="%s.jpg" % filename)
398        if reopened:
399            self.portal_workflow.doActionFor(app,'close')
400        return "successfully loaded passport picture"
401    ###)
402
403    security.declareProtected(ModifyPortalContent,'createOne') ###(
404    def createOne(self,students_folder,student_brain,letter,commit=False):
405        sid = self.waeup_tool.generateStudentId(letter)
406        students_folder.invokeFactory('Student', sid)
407        student = getattr(students_folder,sid)
408        self.portal_workflow.doActionFor(student,'return')
409        student.manage_setLocalRoles(sid, ['Owner',])
410        matric_no = student_brain.matric_no
411        jamb_reg_no = student_brain.Entryregno
412        self.students_catalog.addRecord(id = sid,
413                                           matric_no = matric_no,
414                                           jamb_reg_no = jamb_reg_no,
415                                           sex = student_brain.Sex == "F",
416                                           name = "%s %s %s" % (student_brain.Firstname,
417                                                                student_brain.Middlename,
418                                                                student_brain.Lastname)
419                                        )
420        if commit:
421            transaction.commit()
422        return sid,jamb_reg_no
423    ###)
424
425    security.declareProtected(ModifyPortalContent,'addStudent') ###(
426    def addStudent(self,dict):
427        students_folder = self.portal_url.getPortalObject().campus.students
428        sid = self.waeup_tool.generateStudentId('?')
429        students_folder.invokeFactory('Student', sid)
430        student_obj = getattr(students_folder,sid)
431        f2t = self.student_field2types
432        #from pdb import set_trace; set_trace()
433        d = {}
434        #d['jamb_sex']  = 'M'
435        #if dict.get('sex'):
436        #    d['jamb_sex']  = 'F'
437
438        entry_session = dict.get('entry_session')
439        if entry_session == self.getSessionId()[-2:]:
440            wfaction = 'admit'
441            wft = 'wf_transition_admit'
442            password = None
443        else:
444            wfaction = 'return'
445            wft = 'wf_transition_return'
446            password = self.generatePassword()
447            self.makeStudentMember(sid,password)
448
449        for pt in f2t.keys():
450            student_obj.invokeFactory(pt,f2t[pt]['id'])
451            sub_obj = getattr(student_obj,f2t[pt]['id'])
452            sub_doc = sub_obj.getContent()
453            #self.portal_workflow.doActionFor(sub_obj,'open',dest_container=sub_obj)
454            d['Title'] = f2t[pt]['title']
455            for field in f2t[pt]['fields']:
456                d[field] = dict.get(field,'')
457            sub_doc.edit(mapping = d)
458            new_state = f2t[pt][wft]
459            if new_state != "remain":
460                self.portal_workflow.doActionFor(sub_obj,new_state,dest_container=sub_obj)
461        self.portal_workflow.doActionFor(student_obj,wfaction)
462        student_obj.manage_setLocalRoles(sid, ['Owner',])
463        return sid,password
464    ###)
465
466    security.declarePublic('getCertificateBrain') ###(
467    def getCertificateBrain(self,cert_id):
468        "do it"
469        res = ZCatalog.searchResults(self.portal_catalog_real,
470                                {'portal_type':"Certificate",
471                                      'id': cert_id})
472        if res:
473            return res[0]
474        return None
475    ###)
476
477    security.declareProtected(ModifyPortalContent,'get_csv_filenames') ###(
478    def get_csv_filenames(self):
479        "do it"
480        files = [file for file in os.listdir("%s/import/" % (i_home))
481                 if file.endswith('.csv') and file.find('imported') == -1]
482        return files
483    ###)
484
485    security.declarePublic('findStudentByMatricelNo') ###(
486    def findStudentByMatricelNo(self,matric_no):
487        "do it"
488        res = ZCatalog.searchResults(self.portal_catalog_real,
489                                {'portal_type':"StudentClearance",
490                                 'SearchableText': matric_no})
491        if res:
492            return res[0]
493        return None
494    ###)
495
496    security.declarePublic('makeStudentMember') ###(
497    def makeStudentMember(self,sid,password='uNsEt'):
498        """make the student a member"""
499        membership = self.portal_membership
500        membership.addMember(sid,
501                             password ,
502                             roles=('Member',
503                                     'Student',
504                                     ),
505                             domains='',
506                             properties = {'memberareaCreationFlag': False,
507                                           'homeless': True},)
508        member = membership.getMemberById(sid)
509        self.portal_registration.afterAdd(member, sid, password, None)
510        #self.manage_setLocalRoles(sid, ['Owner',])
511    ###)
512
513    security.declarePublic('makeStudentData') ###(
514    def makeStudentData(self,student_id,email=None,phone_nr=None):
515        "create Datastructure for a returning Student"
516        #import pdb;pdb.set_trace()
517        logger = logging.getLogger('WAeUPTool.makeStudentData')
518        students_folder = self.portal_url.getPortalObject().campus.students
519        #res = self.students_catalog(id=student_id)
520        #if res:
521        #    st = res[0]
522        #res = self.returning_import(matric_no = st.matric_no)
523        res = self.returning_import(id = student_id)
524        if res:
525            student = res[0]
526        else:
527            logger.info('Id %s not found in returning_import' % student_id)
528            return
529        logger.info('%s creates data structure' % student_id)
530        s_results = self.results_import(matric_no = student.matric_no)
531        if s_results:
532            lnr = self.getLevelFromResultsCosCode(s_results)
533            level = "%d00" % lnr
534            verdict,eligible = self.getVerdict(s_results[0].Verdict)
535            if eligible:
536                level = "%d00" % (lnr + 1)
537        else:
538            logger.info('matric_no %s not found in results_import' % student.matric_no)
539            level = 0
540            verdict = 'N/A'
541        #student should not be allowed to perform this transition
542        #wftool = self.portal_workflow
543        #wftool.doActionFor(student,'return')
544        certcode_org = student.Coursemajorcode
545        certcode = makeCertificateCode(certcode_org)
546        certificate_brain = self.getCertificateBrain(certcode)
547        if not certificate_brain:
548            em = 'Certificate %s org-code %s not found\n' % (certcode, certcode_org)
549            logger.info(em)
550        matric_no = student.matric_no
551        sid = student_id
552        student_obj = getattr(students_folder,sid)
553        student_obj.invokeFactory('StudentApplication','application')
554        application = student_obj.application
555        self.portal_workflow.doActionFor(application,'open',dest_container=application)
556        da = {'Title': 'Application Data'}
557        student_obj.invokeFactory('StudentPersonal','personal')
558        da['jamb_reg_no'] = student.Entryregno
559        em = self.getEntryMode(student.Entryregno)
560##        em = student.Mode_of_Entry
561##        if em in ('DIRECT', 'DIRECT ENTRY',):
562##            em = 'DE'
563##        elif em in ('U.M.E', 'UNE',):
564##            em = 'UME'
565##        elif not em:
566##            em = "unknown"
567        da['entry_mode'] = em
568        personal = student_obj.personal
569        self.portal_workflow.doActionFor(personal,'open',dest_container=personal)
570        dp = {'Title': 'Personal Data'}
571        student_obj.invokeFactory('StudentClearance','clearance')
572        clearance = student_obj.clearance
573        self.portal_workflow.doActionFor(clearance,'open',dest_container=clearance)
574        dc = {'Title': 'Clearance/Eligibility Record'}
575        dc['matric_no'] = matric_no
576        state = student.State
577        lga = student.LGA
578        if state and lga:
579            lga =  state + ' / ' + lga
580        else:
581            lga = "None"
582        da['jamb_lga'] = dc['lga'] = lga
583        da['app_email'] = dp['email'] = email
584        da['app_mobile'] = dp['phone'] = phone_nr
585        dp['firstname'] = student.Firstname
586        dp['middlename'] = student.Middlename
587        dp['lastname'] = student.Lastname
588        da['jamb_lastname'] = "%s %s %s" % (student.Firstname,student.Middlename,student.Lastname)
589        da['jamb_sex'] = student.Sex
590        dp['sex'] = student.Sex == 'F'
591        dp['perm_address'] = student.Permanent_Address
592        application.getContent().edit(mapping=da)
593        self.portal_workflow.doActionFor(application,'close',dest_container=application)
594        personal.getContent().edit(mapping=dp)
595        clearance.getContent().edit(mapping=dc)
596        self.portal_workflow.doActionFor(clearance,'close',dest_container=clearance)
597##        catd = {}
598##        catd['id'] = sid
599##        catd['entry_mode']= da['entry_mode']
600##        catd['matric_no'] = matric_no
601##        catd['jamb_reg_no'] = da['jamb_reg_no']
602##        catd['name'] = "%(firstname)s %(middlename)s %(lastname)s" % dp
603##        catd['sex'] = dp['sex']
604##        catd['level'] = level
605##        catd['verdict'] = verdict
606##        if certificate_brain:
607##            cpath = certificate_brain.getPath().split('/')
608##            catd['faculty'] = cpath[-4]
609##            catd['department'] = cpath[-3]
610##            catd['course'] = certcode
611##        self.students_catalog.modifyRecord(**catd)
612        #
613        # Study Course
614        #
615        student_obj.invokeFactory('StudentStudyCourse','study_course')
616        studycourse = student_obj.study_course
617        self.portal_workflow.doActionFor(studycourse,'open',dest_container=studycourse)
618        dsc = {}
619        dsc['study_course'] = certcode
620        dsc['current_level'] = level
621        dsc['current_verdict'] = verdict
622        dsc['current_mode'] = em
623        dsc['current_session'] = '06'
624        studycourse.getContent().edit(mapping=dsc)
625        #
626        # Level
627        #
628##        l = getattr(studycourse,level,None)
629##        if l is None:
630##            studycourse.invokeFactory('StudentStudyLevel', level)
631##            l = getattr(studycourse, level)
632##            self.portal_workflow.doActionFor(l,'open',dest_container=l)
633##            l.getContent().edit(mapping={'Title': "Level %s" % level})
634###)
635
636    security.declarePublic('makeStudentLevel') ###(
637    def makeStudentLevel(self,student_id):
638        "create the StudyLevel for a returning Student"
639        #import pdb;pdb.set_trace()
640        logger = logging.getLogger('WAeUPTool.makeStudentLevel')
641        students_folder = self.portal_url.getPortalObject().campus.students
642        res = self.students_catalog(id=student_id)
643        if res:
644            st = res[0]
645        course = st.course
646        matric_no = st.matric_no
647        level = st.level
648        res = self.results_import(matric_no = matric_no)
649        if res:
650            results = res
651        logger.info('%s creating Level %s' % (student_id,level))
652        #
653        # Level
654        #
655        student_obj = getattr(self.portal_url.getPortalObject().campus.students,student_id)
656        studycourse = getattr(student_obj,"study_course",None)
657        self.portal_workflow.doActionFor(studycourse,'close_for_edit',dest_container=studycourse)
658        l = getattr(studycourse,level,None)
659        if l is None:
660            studycourse.invokeFactory('StudentStudyLevel', level)
661            l = getattr(studycourse, level)
662            self.portal_workflow.doActionFor(l,'open',dest_container=l)
663            l.getContent().edit(mapping={'Title': "Level %s" % level})
664        ###)
665
666    security.declarePublic('getHallInfo') ###(
667    def getHallInfo(self,bed):
668        """return Hall Info"""
669        info = {}
670        hall,block,room,letter = bed.split('_')
671        res = ZCatalog.searchResults(self.portal_catalog_real,portal_type="AccoHall",id=hall)
672        if res and len(res) == 1:
673            hall_brain = res[0]
674            hall_doc = hall_brain.getObject().getContent()
675        else:
676            return info
677        info['hall_title'] = hall_brain.Title
678        info['maintenance_code'] = hall_doc.maintenance_code
679        res = ZCatalog.searchResults(self.portal_catalog_real,portal_type="ScratchCardBatch")
680        batch_doc = None
681        for brain in res:
682            if brain.id.startswith(info['maintenance_code']):
683                batch_doc = brain.getObject().getContent()
684                break
685        if batch_doc is None:
686            info['maintenance_fee'] = None
687        else:
688            info['maintenance_fee'] = batch_doc.cost
689        return info
690    ###)
691
692    security.declarePublic('showFsPicture') ###(
693    def showFsPicture(self,path):
694        """return a picture from the filesystem"""
695        picture_path = "%s/import/%s" % (i_home,path)
696        if os.path.exists(picture_path):
697            return open(picture_path).read()
698
699    ###)
700
701    security.declareProtected(ModifyPortalContent,'deleteAllCourses') ###(
702    def deleteAllCourses(self,department="All"):
703        ''' delete the courses'''
704        pm = self.portal_membership
705        member = pm.getAuthenticatedMember()
706
707        if str(member) not in ("henrik","joachim"):
708            return "not possible"
709        if department == "All":
710            res = self.portal_catalog({'meta_type': 'Department'})
711        if len(res) < 1:
712            return "No Departments found"
713
714        deleted = []
715        for dep in res:
716            cf = dep.getObject().courses
717            if cf:
718                cf.manage_delObjects(ids=cf.objectIds())
719                deleted.append("deleted Courses in %s" % dep.getId)
720        return "\r".join(deleted)
721    ###)
722
723    security.declareProtected(ModifyPortalContent,'getLogfileLines') ###(
724    def getLogfileLines(self,filename="event.log",numlines=20):
725        """Get last NUMLINES lines of logfile FILENAME.
726
727        Return last lines' of a file in the instances logfile directory as
728        a list. The number of returned lines equals `numlines' or less. If
729        less than `numlines' lines are available, the whole file ist
730        returned. If the file can not be opened or some other error
731        occurs, empty list is returend.
732        """
733        result = []
734        lines_hit = 0
735
736        # We only handle files in instances' log directory...
737        logpath = os.path.join(i_home, "log")
738        filename = str(os.path.abspath( os.path.join( logpath, filename )))
739        if not filename.startswith( logpath ):
740            # Attempt to access file outside log-dir...
741            return []
742
743        try:
744            fd = file( filename, "rb" )
745        except IOError:
746            return []
747        if not fd:
748            return []
749
750        if os.linesep == None:
751            linesep = '\n'
752        else:
753            linesep = os.linesep
754
755        # Try to find 'numlines' times a lineseparator, searching from end
756        # and moving to the beginning of file...
757        fd.seek( 0, 2) # Move to end of file...
758        while lines_hit < numlines:
759            if fd.read(1) == linesep[-1]: # This moves filedescriptor
760                                          # one step forward...
761                lines_hit += 1
762            try:
763                fd.seek( -2, 1) # Go two bytes back from current pos...
764            except IOError:
765                # We cannot go back two bytes. Maybe the file is too small...
766                break
767        fd.seek(2,1)
768
769        # Read all lines from current position...
770        result = fd.readlines()
771        # Remove line endings...
772        result = [x.strip() for x in result]
773        fd.close()
774        return result
775    ###)
776
777    security.declareProtected(ModifyPortalContent,"getCallbacksFromLog")###(
778    def getCallbacksFromLog(self,filename):
779        """fix Online Payment Transactions from Z2.log entries"""
780        import transaction
781        import random
782        from cgi import parse_qs
783        from urlparse import urlparse
784        #from pdb import set_trace
785        wftool = self.portal_workflow
786        current = DateTime.DateTime().strftime("%d-%m-%y_%H_%M_%S")
787        students_folder = self.portal_url.getPortalObject().campus.students
788        s = r'(?P<client_ip>\S+) - (?P<member_id>\S+) \['
789        s += r'(?P<date>.*)\] "(?P<get>.*)" (?P<codes>\d+ \d+) "'
790        s += r'(?P<intersw>.*)" "(?P<agent>.*)"'
791        data = re.compile(s)
792        start = True
793        tr_count = 1
794        total = 0
795        #name = 'pume_results'
796        #name = 'epaymentsuccessful_z2log2'
797        name = filename
798        no_import = []
799        imported = []
800        logger = logging.getLogger('WAeUPTool.getFailedTransactions')
801        try:
802            transactions = open("%s/import/%s" % (i_home,name),"rb").readlines()
803        except:
804            logger.error('Error reading %s' % name)
805            return
806        tas = []
807        for line in transactions:
808            dict = {}
809            items = data.search(line)
810            dict['idict'] = idict = items.groupdict()
811            #print idict
812            #from pdb import set_trace;set_trace()
813            urlparsed = urlparse(idict['get'][4:])
814            #print urlparsed
815            path = urlparsed[2].split('/')
816            dict['student_id'] = student_id = path[8]
817            dict['payment_id'] = payment_id = path[10]
818            dict['qs_dict'] = qs_dict = parse_qs(urlparsed[4])
819            tas.append(dict)
820            tr_count += 1
821        return tas
822    ###)
823
824    security.declareProtected(ModifyPortalContent,"importOnlinePaymentTransactions")###(
825    def importOnlinePaymentTransactions(self):
826        """load Online Payment Transactions from CSV values"""
827        import transaction
828        import random
829        #from pdb import set_trace
830        current = DateTime.DateTime().strftime("%d-%m-%y_%H_%M_%S")
831        opt = self.online_payments_import
832        students_folder = self.portal_url.getPortalObject().campus.students
833        start = True
834        tr_count = 1
835        total = 0
836        #name = 'pume_results'
837        name = 'OnlineTransactions'
838        no_import = []
839        imported = []
840        logger = logging.getLogger('WAeUPTool.importOnlinePaymentTransactions')
841        try:
842            transactions = csv.DictReader(open("%s/import/%s.csv" % (i_home,name),"rb"))
843        except:
844            logger.error('Error reading %s.csv' % name)
845            return
846        for pay_transaction in transactions:
847            if start:
848                start = False
849                logger.info('Start loading from %s.csv' % name)
850                s = ','.join(['"%s"' % fn for fn in pay_transaction.keys()])
851                no_import.append('%s,"Error"' % s)
852                format = ','.join(['"%%(%s)s"' % fn for fn in pay_transaction.keys()])
853                format_error = format + ',"%(Error)s"'
854            data = {}
855
856            # format of the first file sent by Tayo
857            #data['datetime'] = date = DateTime.DateTime(pay_transaction['Date'])
858            #data['student_id'] = student_id = pay_transaction['Payer ID']
859            #data['order_id'] = order_id = pay_transaction['Order ID (Tranx Ref)']
860            #data['response_code'] = response_code = pay_transaction['Resp Code']
861            #data['amount'] = amount = pay_transaction['Amount']
862
863            # format of the second file sent by Tayo
864            #data['datetime'] = date = 0
865            #data['student_id'] = student_id = pay_transaction['Payer ID']
866            #data['order_id'] = order_id = pay_transaction['Order ID (Tranx Ref)']
867            #data['response_code'] = response_code = '00'
868            #data['amount'] = amount = pay_transaction['Amount']
869
870            # format of the third file sent by Kehinde
871            data['datetime'] = date = 0
872            data['student_id'] = student_id = pay_transaction['customer_id']
873            data['order_id'] = order_id = pay_transaction['merchant_reference']
874            data['response_code'] = response_code = '00'
875            data['amount'] = amount = pay_transaction['Amount']
876
877            dup = False
878            if response_code == "12":
879                continue
880            try:
881                opt.addRecord(**data)
882            except ValueError:
883                dup = True
884            #from pdb import set_trace;set_trace()
885            if dup:
886                if response_code == "00":
887                    try:
888                        opt.modifyRecord(**data)
889                    except:
890                        logger.info("duplicate uid, order_id %(order_id)s, student_id %(student_id)s, response_code %(response_code)s" % data)
891                        continue
892                else:
893                    pay_transaction['Error'] = "Duplicate order_id"
894                    no_import.append( format_error % pay_transaction)
895                    logger.info("duplicate order_id %(order_id)s for %(student_id)s %(response_code)s" % data)
896                    continue
897            tr_count += 1
898            if tr_count > 1000:
899                if len(no_import) > 0:
900                    open("%s/import/%s_not_imported%s.csv" % (i_home,name,current),"a").write(
901                             '\n'.join(no_import) + '\n')
902                    no_import = []
903                em = '%d transactions committed\n' % (tr_count)
904                transaction.commit()
905                regs = []
906                logger.info(em)
907                total += tr_count
908                tr_count = 0
909        open("%s/import/%s_not_imported%s.csv" % (i_home,name,current),"a").write(
910                                                '\n'.join(no_import))
911        return self.REQUEST.RESPONSE.redirect("%s" % self.REQUEST.get('URL1'))
912    ###)
913
914    security.declareProtected(ModifyPortalContent,'mass_create_faculty') ###(
915    def mass_create_faculty(self,mapping):
916        "create a faculty"
917        logger = logging.getLogger('WAeUPTool.mass_create_faculty')
918        academics_folder = self.portal_url.getPortalObject().campus.academics
919        fid = mapping['code']
920        if getattr(academics_folder,fid,None) is not None:
921            return '', "Faculty with ID: %s exists" % fid
922        logger.info('Creating Faculty %(code)s, %(title)s' % mapping)
923        try:
924            academics_folder.invokeFactory('Faculty', fid)
925        except BadRequest,E:
926            return '', "%s" % E
927        f = getattr(academics_folder,fid,None)
928        f.getContent().edit(mapping=mapping)
929        return fid,''
930    ###)
931
932    security.declareProtected(ModifyPortalContent,'mass_edit_faculty') ###(
933    def mass_edit_faculty(self,mapping):
934        "edit a faculty"
935        logger = logging.getLogger('WAeUPTool.mass_edit_faculty')
936        academics_folder = self.portal_url.getPortalObject().campus.academics
937        fid = mapping['code']
938        f = getattr(academics_folder,fid,None)
939        if f is None:
940            return '', "Faculty with ID: %s does not exist" % fid
941        logger.info('Editing Faculty %(code)s, %(title)s' % mapping)
942        f.getContent().edit(mapping=mapping)
943        return fid,''
944    ###)
945
946    security.declareProtected(ModifyPortalContent,'mass_create_department') ###(
947    def mass_create_department(self,mapping):
948        "create a department in the correct faculty"
949        logger = logging.getLogger('WAeUPTool.mass_create_department')
950        fid = mapping['faculty_code']
951        if getattr(self,'_v_faculties',None) is None:
952            res = self.portal_catalog(portal_type = "Faculty")
953            self._v_faculties = {}
954            for f in res:
955                self._v_faculties[f.getId] = f.getObject()
956        f = self._v_faculties.get(fid,None)
957        if f is None:
958            return '', "No Faculty with ID: %s" % fid
959        else:
960            did = mapping.get('code')
961            d = getattr(f,did,None)
962            if d is None or d.portal_type == "Faculty":
963                logger.info('Creating Department %(code)s, %(title)s' % mapping)
964                try:
965                    f.invokeFactory('Department', did)
966                except BadRequest,E:
967                    return '', "%s" % E
968                d = getattr(f,did)
969                d.invokeFactory('CoursesFolder','courses')
970                courses = getattr(d,'courses')
971                dict = {'Title': 'Courses'}
972                courses.getContent().edit(mapping=dict)
973                d.invokeFactory('CertificatesFolder','certificates')
974                certificates = getattr(d,'certificates')
975                dict = {'Title': 'Certificates'}
976                certificates.getContent().edit(mapping=dict)
977            d.getContent().edit(mapping=mapping)
978        return did,''
979    ###)
980
981    security.declareProtected(ModifyPortalContent,'mass_edit_department') ###(
982    def mass_edit_department(self,mapping):
983        "create a department in the correct faculty"
984        logger = logging.getLogger('WAeUPTool.mass_create_department')
985        academics_folder = self.portal_url.getPortalObject().campus.academics
986        fid = mapping['faculty_code']
987        did = mapping.get('code')
988        try:
989            d = getattr(getattr(academics_folder,fid),did,None)
990        except KeyError:
991            return '', "Department %s or Faculty %s wrong" % (did,fid)
992        else:
993            if d is None or d.portal_type == "Faculty":
994                logger.info('Editing Department %(code)s, %(title)s' % mapping)
995            d.getContent().edit(mapping=mapping)
996        return did,''
997    ###)
998
999    security.declareProtected(ModifyPortalContent,'mass_create_course') ###(
1000    def mass_create_course(self,mapping):
1001        #import pdb;pdb.set_trace()
1002        if getattr(self,'_v_course_list',None) is None:
1003            self._v_course_list = []
1004        if getattr(self,'_v_departments',None) is None:
1005            res = self.portal_catalog(portal_type = "Department")
1006            self._v_department_courses = {}
1007            for d in res:
1008                self._v_department_courses[d.getId] = getattr(d.getObject(),"courses",None)
1009        did = mapping['department_code']
1010        d = self._v_department_courses.get(did,None)
1011        if d is None:
1012            return '', "No Department with ID: %s" % did
1013        course_id = mapping.get('code')
1014        if course_id in self._v_course_list:
1015            return '', "Duplicate Course ID: %s" % did
1016        c = getattr(d,course_id,None)
1017        if c is not None:
1018            return '', "Duplicate Course ID: %s" % did
1019        try:
1020            d.invokeFactory('Course', course_id)
1021        except BadRequest,E:
1022            return '', "%s" % E
1023        self._v_course_list.append(course_id)
1024        c = getattr(d,course_id)
1025        c.getContent().edit(mapping=mapping)
1026        return course_id,''
1027    ###)
1028
1029    security.declareProtected(ModifyPortalContent,'mass_edit_course') ###(
1030    def mass_edit_course(self,mapping):
1031        #import pdb;pdb.set_trace()
1032        course_id = mapping.get('code')
1033        res = self.portal_catalog(id=course_id)
1034        if not res:
1035            return '', "No Course with ID: %s" % course_id
1036        c = res[0].getObject()
1037        c.getContent().edit(mapping=mapping)
1038        return course_id,''
1039    ###)
1040
1041    security.declareProtected(ModifyPortalContent,'mass_create_certificate') ###(
1042    def mass_create_certificate(self,mapping):
1043        if getattr(self,'_v_certificate_list',None) is None:
1044            self._v_certificate_list = []
1045        if getattr(self,'_v_department_certificates',None) is None:
1046            res = self.portal_catalog(portal_type = "Department")
1047            self._v_department_certificates = {}
1048            for d in res:
1049                self._v_department_certificates[d.getId] = getattr(d.getObject(),"certificates",None)
1050        did = mapping['department_code']
1051        d = self._v_department_certificates.get(did,None)
1052        if d is None:
1053            return '', "No Department with ID: %s" % did
1054        certificate_id = mapping.get('code')
1055        if certificate_id in self._v_certificate_list:
1056            return '', "Duplicate Certificate ID: %s" % did
1057        c = getattr(d,certificate_id,None)
1058        if c is not None:
1059            return '', "Duplicate Certificate ID: %s" % did
1060        try:
1061            d.invokeFactory('Certificate', certificate_id)
1062        except BadRequest,E:
1063            return '', "%s" % E
1064        self._v_certificate_list.append(certificate_id)
1065        c = getattr(d,certificate_id)
1066        c.getContent().edit(mapping=mapping)
1067        return certificate_id,''
1068    ###)
1069
1070    security.declareProtected(ModifyPortalContent,'mass_edit_certificate') ###(
1071    def mass_edit_certificate(self,mapping):
1072        #import pdb;pdb.set_trace()
1073        certificate_id = mapping.get('code')
1074        res = self.portal_catalog(id=certificate_id)
1075        if not res:
1076            return '', "No Certificate with ID: %s" % did
1077        c = res[0].getObject()
1078        c.getContent().edit(mapping=mapping)
1079        return certificate_id,''
1080    ###)
1081
1082    security.declareProtected(ModifyPortalContent,'mass_create_certificate_course') ###(
1083    def mass_create_certificate_course(self,mapping):
1084        if getattr(self,'_v_certificate_course_list',None) is None:
1085            self._v_certificate_course_list = []
1086        if getattr(self,'_v_ceritficates',None) is None:
1087            res = self.portal_catalog(portal_type = "Certificate")
1088            self._v_certificates = {}
1089            for cert in res:
1090                self._v_certificates[cert.getId] = cert.getObject()
1091        certificate_course_id = mapping.get('code')
1092        if not self.courses_catalog(code = certificate_course_id):
1093            return '', "No Course with ID: %s" % certificate_course_id
1094        cert_id = mapping['certificate_code']
1095        cert = self._v_certificates.get(cert_id,None)
1096        if cert is None:
1097            return '', "No Certificate with ID: %s" % cert_id
1098        if certificate_course_id in self._v_certificate_course_list:
1099            return '', "Duplicate CertificateCourse ID: %s" % certificate_course_id
1100        level_id = mapping.get('level')
1101        level = getattr(cert,level_id,None)
1102        if level is None:
1103            cert.invokeFactory('StudyLevel', level_id)
1104            level = getattr(cert,level_id,None)
1105        elif hasattr(level,certificate_course_id):
1106            return '', "Duplicate CertificateCourse ID: %s in %s/ %s" %\
1107            (certificate_course_id,cert_id,level_id)
1108        level.invokeFactory('CertificateCourse', certificate_course_id)
1109        self._v_certificate_course_list.append(certificate_course_id)
1110        c = getattr(level,certificate_course_id)
1111        c.getContent().edit(mapping=mapping)
1112        return certificate_course_id,''
1113    ###)
1114
1115    field2types_student = {   ###(
1116                      'StudentApplication':
1117                          {'id': 'application',
1118                           'title': 'Application Data',
1119                           'wf_transition_return': 'close',
1120                           'wf_transition_admit': 'remain',
1121                           'fields':
1122                             ('jamb_reg_no',
1123                              'entry_mode',
1124                              'entry_session',
1125                              'jamb_score',
1126                              'app_email',
1127                              'jamb_age',
1128                              #'jamb_state',
1129                              #'jamb_lga',
1130                              )
1131                              },
1132                      #'StudentPume':
1133                      #    {'id': 'pume',
1134                      #     'title': 'Pume Data',
1135                      #     'wf_transition_return': 'close',
1136                      #     'wf_transition_admit': 'close',
1137                      #     'fields':
1138                      #       ('pume_score',
1139                      #        )
1140                      #        },
1141                      'StudentClearance':
1142                          {'id': 'clearance',
1143                           'title': 'Clearance/Eligibility Record',
1144                           'wf_transition_return': 'close',
1145                           'wf_transition_admit': 'remain',
1146                           'fields':
1147                             ('matric_no',
1148                              'nationality',
1149                              'lga',
1150                              'birthday',
1151                              )
1152                              },
1153                         'StudentPersonal':
1154                          {'id': 'personal',
1155                           'title': 'Personal Data',
1156                           'wf_transition_return': 'open',
1157                           'wf_transition_admit': 'remain',
1158                           'fields':
1159                             ('firstname',
1160                              'middlename',
1161                              'lastname',
1162                              'sex',
1163                              'email',
1164                              'phone',
1165                              'perm_address',
1166                              )
1167                              },
1168                         'StudentStudyCourse':
1169                          {'id': 'study_course',
1170                           'title': 'Study Course',
1171                           'wf_transition_return': 'open',
1172                           'wf_transition_admit': 'remain',
1173                           'fields':
1174                             ('study_course',
1175                              'current_level',
1176                              'current_session',
1177                              'current_mode',
1178                              'current_verdict',
1179                              )
1180                              },
1181
1182                         'PaymentsFolder':
1183                          {'id': 'payments',
1184                           'title': 'Payments',
1185                           'wf_transition_return': 'open',
1186                           'wf_transition_admit': 'open',
1187                           'fields':
1188                             ()
1189                              },
1190                         }
1191    ###)
1192
1193    security.declareProtected(ModifyPortalContent,'mass_create_student') ###(
1194    def mass_create_student(self,mapping):
1195        "create a students record due import"
1196        logger = logging.getLogger('WAeUPTool.mass_create_student')
1197        students_folder = self.portal_url.getPortalObject().campus.students
1198        jamb_reg_no = mapping.get('jamb_reg_no',None)
1199        if jamb_reg_no:
1200            res = self.students_catalog(jamb_reg_no = jamb_reg_no)
1201            if res:
1202                return '',"jamb_reg_no exists"
1203        matric_no = mapping.get('matric_no',None)
1204        if matric_no:
1205            res = self.students_catalog(matric_no = matric_no)
1206            if res:
1207                return '',"matric_no exists"
1208        sid = self.waeup_tool.generateStudentId('?')
1209        students_folder.invokeFactory('Student', sid)
1210        student_obj = getattr(students_folder,sid)
1211        f2t = self.field2types_student
1212        d = {}
1213        d['jamb_sex']  = 'M'
1214        if mapping.get('sex'):
1215            d['jamb_sex']  = 'F'
1216        transition = mapping.get('reg_transition','admit')
1217        if transition not in ('admit','return'):
1218            return '',"no valid transition provided"
1219        for pt in f2t.keys():
1220            student_obj.invokeFactory(pt,f2t[pt]['id'])
1221            sub_obj = getattr(student_obj,f2t[pt]['id'])
1222            sub_doc = sub_obj.getContent()
1223            d['Title'] = f2t[pt]['title']
1224            for field in f2t[pt]['fields']:
1225                d[field] = mapping.get(field,'')
1226            if pt == "StudyCourse":
1227                for von,zu in (('entry_mode','current_mode'),
1228                               ('entry_session','current_session')):
1229                    if mapping.get(zu,None) is None and mapping.get(von,None) is not None:
1230                        d[zu] = mapping[von]
1231            sub_doc.edit(mapping = d)
1232
1233            #import pdb;pdb.set_trace()
1234            new_state = f2t[pt]['wf_transition_%(transition)s' % vars()]
1235            if new_state != "remain":
1236                self.portal_workflow.doActionFor(sub_obj,new_state,dest_container=sub_obj)
1237        self.portal_workflow.doActionFor(student_obj,transition)
1238        student_obj.manage_setLocalRoles(sid, ['Owner',])
1239        return sid,''
1240    ###)
1241
1242    security.declareProtected(ModifyPortalContent,'mass_edit_student') ###(
1243    def mass_edit_student(self,mapping):
1244        "edit a students record due import"
1245        logger = logging.getLogger('WAeUPTool.mass_edit_student')
1246        students_folder = self.portal_url.getPortalObject().campus.students
1247        sid = mapping.get('id',None)
1248        jamb_reg_no = mapping.get('jamb_reg_no',None)
1249        matric_no = mapping.get('matric_no',None)
1250        editable_keys = mapping.keys()
1251        if sid:
1252            res = self.students_catalog(id = sid)
1253            if not res:
1254                return '',"no student with id %s" % sid
1255            if matric_no and res[0].matric_no and\
1256              matric_no != res[0].matric_no:
1257                logger.info("%s, old matric_no %s overwritten with %s" % (res[0].id,res[0].matric_no,matric_no))
1258            if jamb_reg_no and res[0].jamb_reg_no and\
1259              jamb_reg_no != res[0].jamb_reg_no:
1260                logger.info("%s, old reg_no %s overwritten with %s" % (res[0].id,res[0].jamb_reg_no,jamb_reg_no))
1261        elif jamb_reg_no:
1262            res = self.students_catalog(jamb_reg_no = jamb_reg_no)
1263            if not res:
1264                return '',"no student with jamb_reg_no %s" % jamb_reg_no
1265            editable_keys.remove('jamb_reg_no')
1266        elif matric_no:
1267            res = self.students_catalog(matric_no = matric_no)
1268            if not res:
1269                return '',"no student with matric_no %s" % matric_no
1270            editable_keys.remove('matric_no')
1271
1272        ## included only to change wf state from admitted to returning
1273        #if res[0].review_state not in ('admitted','objection_raised'):
1274        #    return '%s' % res[0].id ,"student is not in state admitted or objection_raised"
1275        ## end inclusion
1276
1277        sid = res[0].id
1278        student_obj = getattr(students_folder,sid)
1279        f2t = self.field2types_student
1280        d = {}
1281        #import pdb;pdb.set_trace()
1282        any_change = False
1283        for pt in f2t.keys():
1284            if pt == "student_application":
1285                d['jamb_sex']  = 'M'
1286                if mapping.get('sex'):
1287                    d['jamb_sex']  = 'F'
1288            intersect = set(f2t[pt]['fields']).intersection(set(editable_keys))
1289            if intersect:
1290                sub_obj = getattr(student_obj,f2t[pt]['id'],None)
1291                if sub_obj is None:
1292                    try:
1293                        student_obj.invokeFactory(pt,f2t[pt]['id'])
1294                    except:
1295                        continue
1296                    sub_obj = getattr(student_obj,f2t[pt]['id'])
1297                    d['Title'] = f2t[pt]['title']
1298                sub_doc = sub_obj.getContent()
1299                for field in intersect:
1300                    changed = False
1301                    if getattr(sub_doc,field) != mapping.get(field,''):
1302                        any_change = True
1303                        changed = True
1304                        d[field] = mapping.get(field,'')
1305                    if changed:
1306                        sub_doc.edit(mapping = d)
1307
1308
1309        ## included only to change wf state from admitted to returning
1310        #    if res[0].review_state in ('admitted','objection_raised'):
1311        #        new_state = f2t[pt]['wf_transition_return']
1312        #        sub_obj = getattr(student_obj,f2t[pt]['id'],None)
1313        #        if sub_obj and new_state != "remain":
1314        #            try:
1315        #                self.portal_workflow.doActionFor(sub_obj,new_state,dest_container=sub_obj)
1316        #            except:
1317        #                #logger.info('%s, wf transition %s of %s failed' % (sid,new_state,sub_obj.id))
1318        #                pass
1319        #if res[0].review_state in ('admitted','objection_raised'):
1320        #    wfaction = 'return'
1321        #    try:
1322        #        self.portal_workflow.doActionFor(student_obj,wfaction)
1323        #        logger.info('%s, wf state changed' % sid)
1324        #        any_change = True
1325        #    except:
1326        #        logger.info('%s, wf transition failed, old state = %s' % (sid,res[0].review_state))
1327        #        pass
1328        ## end inclusion
1329
1330
1331        if any_change:
1332            return sid,''
1333        else:
1334            return sid,'not modified'
1335    ###)
1336
1337    security.declareProtected(ModifyPortalContent,"importData")###(
1338    def importData(self,filename,name,edit=False,bypass_queue_catalog=False):
1339        """load data from CSV values"""
1340        import transaction
1341        import random
1342
1343        pm = self.portal_membership
1344        member = pm.getAuthenticatedMember()
1345
1346        logger = logging.getLogger('WAeUPTool.importData')
1347        current = DateTime.DateTime().strftime("%d-%m-%y_%H_%M_%S")
1348        students_folder = self.portal_url.getPortalObject().campus.students
1349        start = True
1350        tr_count = 1
1351        total_imported = 0
1352        total_not_imported = 0
1353        total = 0
1354        iname = "import_%s" % name
1355        stool = getToolByName(self, 'portal_schemas')
1356        ltool = getToolByName(self, 'portal_layouts')
1357        schema = stool._getOb(iname)
1358        if schema is None:
1359            em = 'No such schema %s' % iname
1360            logger.error('No such schema %s' % iname)
1361            return em
1362        layout = ltool._getOb(iname)
1363        if layout is None:
1364            em = 'No such layout %s' % iname
1365            logger.error(em)
1366            return em
1367        validators = {}
1368        for widget in layout.keys():
1369            validators[widget] = layout[widget].validate
1370        #import pdb;pdb.set_trace()
1371        mode = "create"
1372        if edit:
1373            mode = "edit"
1374        importer_name = "mass_%(mode)s_%(name)s" % vars()
1375        importer = getattr(self, '%s' % importer_name,None)
1376        if importer is None:
1377            em = 'No importer function %s' % importer_name
1378            logger.error(em)
1379            return em
1380        not_imported = []
1381        imported = []
1382        try:
1383            items = csv.DictReader(open("%s/import/%s.csv" % (i_home,filename),"rb"))
1384        except:
1385            em = 'Error reading %s.csv' % filename
1386            logger.error(em)
1387            return em
1388        for item in items:
1389            if start:
1390                start = False
1391                adapters = [MappingStorageAdapter(schema, item)]
1392                logger.info('%s starts import from %s.csv in %s mode with schema and layout %s' % (member,filename,mode,iname))
1393                attrs = csv.reader(open("%s/import/%s.csv" % (i_home,filename),"rb")).next()
1394                import_keys = [k for k in attrs if not k.startswith('ignore')]
1395                diff2schema = set(import_keys).difference(set(schema.keys()))
1396                diff2layout = set(import_keys).difference(set(layout.keys()))
1397                if diff2schema:
1398                    em = 'not ignorable key(s): "%s" found in heading' % ", ".join(diff2schema)
1399                    return em
1400                s = ','.join(['"%s"' % fn for fn in import_keys])
1401                open("%s/import/%s_not_imported%s.csv" % (i_home,filename,current),"a").write(s + ',"Error"'+ '\n')
1402                s = '"id",' + s
1403                open("%s/import/%s_imported%s.csv" % (i_home,filename,current),"a").write(s + '\n')
1404                format = ','.join(['"%%(%s)s"' % fn for fn in import_keys])
1405                format_error = format + ',"%(Error)s"'
1406                format = '"%(id)s",'+ format
1407
1408            dm = DataModel(item, adapters,context=self)
1409            ds = DataStructure(data=item,datamodel=dm)
1410            error_string = ""
1411            for k in import_keys:
1412                #import pdb;pdb.set_trace()
1413                if not validators[k](ds,mode=mode):
1414                    error_string += " %s : %s" % (k,
1415                                                  self.translation_service(ds.getError(k),
1416                                                                           ds.getErrorMapping(k)))
1417            if not error_string:
1418                item.update(dm)
1419                item['id'],error = importer(item)
1420                if error:
1421                    error_string += error
1422            if error_string:
1423                item['Error'] = error_string
1424                not_imported.append(format_error % item)
1425                total_not_imported += 1
1426            else:
1427                em = format % item
1428                imported.append(em)
1429                logger.info("%(total_imported)d of %(total)d %(em)s" % vars())
1430                tr_count += 1
1431                total_imported += 1
1432            total += 1
1433            if total and not total % 100:
1434                transaction.commit()
1435                if len(not_imported) > 0:
1436                    open("%s/import/%s_not_imported%s.csv" % (i_home,filename,current),"a").write(
1437                             '\n'.join(not_imported) + '\n')
1438                    not_imported = []
1439                if len(imported) > 0:
1440                    open("%s/import/%s_imported%s.csv" % (i_home,filename,current),"a").write(
1441                             '\n'.join(imported) + '\n')
1442                    imported = []
1443                em = '%d transactions committed\n' % (tr_count)
1444                regs = []
1445                logger.info(em)
1446                tr_count = 0
1447        if len(imported) > 0:
1448            open("%s/import/%s_imported%s.csv" % (i_home,filename,current),"a").write(
1449                                                '\n'.join(imported))
1450        if len(not_imported) > 0:
1451            open("%s/import/%s_not_imported%s.csv" % (i_home,filename,current),"a").write(
1452                                                '\n'.join(not_imported))
1453        em = "Imported: %d, not imported: %d of total %d" % (total_imported,total_not_imported,total)
1454        return em
1455    ###)
1456
1457    security.declareProtected(ModifyPortalContent,"moveImagesToFS")###(
1458    def moveImagesToFS(self,student_id="O738726"):
1459        "move the images to the filesystem"
1460        images_dir = os.path.join("%s/images" % i_home,student_id)
1461        student_folder = getattr(self.portal_url.getPortalObject().campus.students,student_id)
1462        stool = getToolByName(self, 'portal_schemas')
1463        schemas = ['student_application',
1464                   'student_clearance',
1465                   ]
1466        created = False
1467        for schema_id in schemas:
1468            schema = stool._getOb(schema_id)
1469            object = getattr(student_folder,schema_id[len('student_'):],None)
1470            if object is None:
1471                continue
1472            doc = object.getContent()
1473            for key in schema.keys():
1474                if schema[key].meta_type != "CPS Image Field":
1475                    continue
1476                #import pdb;pdb.set_trace()
1477                image = getattr(doc,key)
1478                if not hasattr(image,"data"):
1479                    continue
1480                if not created:
1481                    if not os.path.exists(images_dir):
1482                        os.mkdir(images_dir)
1483                    created = True
1484                filename = os.path.join(images_dir,"%(key)s_%(student_id)s.jpg" % vars())
1485                open(filename,"wb").write(str(image.data))
1486
1487    security.declareProtected(ModifyPortalContent,"movePassportToFS")###(
1488    def movePassportToFS(self,student_id="O738726"):
1489        "move the passports to the filesystem"
1490        images_dir = os.path.join("%s" % i_home,'passports')
1491        student_folder = getattr(self.portal_url.getPortalObject().campus.students,student_id)
1492        stool = getToolByName(self, 'portal_schemas')
1493        schemas = ['student_application',
1494                   #'student_clearance',
1495                   ]
1496        created = False
1497        for schema_id in schemas:
1498            schema = stool._getOb(schema_id)
1499            object = getattr(student_folder,schema_id[len('student_'):],None)
1500            if object is None:
1501                continue
1502            doc = object.getContent()
1503            for key in schema.keys():
1504                if schema[key].meta_type != "CPS Image Field":
1505                    continue
1506                #import pdb;pdb.set_trace()
1507                image = getattr(doc,key)
1508                if not hasattr(image,"data"):
1509                    continue
1510                if not created:
1511                    if not os.path.exists(images_dir):
1512                        os.mkdir(images_dir)
1513                    created = True
1514                filename = os.path.join(images_dir,"%(student_id)s.jpg" % vars())
1515                open(filename,"wb").write(str(image.data))
1516
1517InitializeClass(WAeUPTool)
Note: See TracBrowser for help on using the repository browser.