Changeset 9265 for main/waeup.kofa/trunk/src/waeup/kofa/accesscodes
- Timestamp:
- 1 Oct 2012, 21:18:48 (12 years ago)
- Location:
- main/waeup.kofa/trunk/src/waeup/kofa/accesscodes
- Files:
-
- 1 added
- 5 edited
Legend:
- Unmodified
- Added
- Removed
-
main/waeup.kofa/trunk/src/waeup/kofa/accesscodes/accesscode.py
r9263 r9265 81 81 grok.implements(IAccessCode) 82 82 83 def __init__(self, batch_serial , random_num):83 def __init__(self, batch_serial=None, random_num=None): 84 84 super(AccessCode, self).__init__() 85 85 self.batch_serial = batch_serial … … 154 154 155 155 def __init__(self, creation_date=None, creator=None, batch_prefix=None, 156 cost=None, entry_num= None, num=None):156 cost=None, entry_num=0, num=None): 157 157 super(AccessCodeBatch, self).__init__() 158 158 self.creation_date = creation_date … … 317 317 318 318 def addBatchByImport(self, batch, batch_id): 319 """Add an already created `batch` by import 320 with defined id. 321 """ 319 """Add an already created `batch` by import with defined id. 320 321 We want to create a batch without access codes. Since num_entry 322 access codes are automatically added by handle_batch_added when 323 the batch is added to the ZODB, we have to temporarily set entry_num 324 to zero when adding the batch persistently. 325 """ 326 orig_entry_num = batch.entry_num 327 batch.entry_num = 0 322 328 self[batch_id] = batch 329 self._p_changed = True 330 batch.entry_num = orig_entry_num 323 331 return 324 332 … … 480 488 481 489 def __call__(self, *args, **kw): 482 return AccessCodeBatch( *args, **kw)490 return AccessCodeBatch() 483 491 484 492 def getInterfaces(self): 485 493 return implementedBy(AccessCodeBatch) 494 495 class AccessCodeFactory(grok.GlobalUtility): 496 """A factory for accesscodes. 497 498 We need this factory for the accesscodeprocessor. 499 """ 500 grok.implements(IFactory) 501 grok.name(u'waeup.AccessCode') 502 title = u"Create a new accesscode.", 503 description = u"This factory instantiates new accesscode instances." 504 505 def __call__(self, *args, **kw): 506 return AccessCode(*args, **kw) 507 508 def getInterfaces(self): 509 return implementedBy(AccessCode) 486 510 487 511 def get_access_code(access_code): -
main/waeup.kofa/trunk/src/waeup/kofa/accesscodes/batching.py
r9263 r9265 20 20 """ 21 21 import grok 22 from hurry.workflow.interfaces import IWorkflowState 22 23 from zope.interface import Interface 23 from waeup.kofa.interfaces import IBatchProcessor, IGNORE_MARKER 24 from waeup.kofa.interfaces import IBatchProcessor, IGNORE_MARKER, IObjectHistory 24 25 from waeup.kofa.utils.batching import BatchProcessor 25 from waeup.kofa.accesscodes.interfaces import IAccessCodeBatch 26 from waeup.kofa.accesscodes.interfaces import IAccessCodeBatch, IAccessCode 26 27 27 28 class AccessCodeBatchProcessor(BatchProcessor): … … 67 68 Returns a string describing the fields changed. 68 69 """ 69 changed = [] 70 for key, value in row.items(): 71 # Skip fields to be ignored. 72 if value == IGNORE_MARKER: 73 continue 74 if not hasattr(obj, key): 75 continue 76 try: 77 evalvalue = eval(value) 78 if isinstance(evalvalue, list): 79 value = eval(value) 80 except: 81 pass 82 setattr(obj, key, value) 83 log_value = getattr(value, 'code', value) 84 changed.append('%s=%s' % (key, log_value)) 85 70 items_changed = super(AccessCodeBatchProcessor, self).updateEntry( 71 obj, row, site) 86 72 # Log actions... 87 73 location_field = self.location_fields[0] 88 items_changed = ', '.join(changed)89 74 grok.getSite()['accesscodes'].logger.info('%s - %s - Batch updated: %s' 90 75 % (self.name, row[location_field], items_changed)) 91 76 return 77 78 class AccessCodeProcessor(BatchProcessor): 79 """A batch processor for IAccessCode objects. 80 """ 81 grok.implements(IBatchProcessor) 82 grok.provides(IBatchProcessor) 83 grok.context(Interface) 84 util_name = 'accesscodeprocessor' 85 grok.name(util_name) 86 87 name = u'AccessCode Processor' 88 iface = IAccessCode 89 90 location_fields = ['representation', 'batch_prefix', 'batch_num'] 91 factory_name = 'waeup.AccessCode' 92 93 mode = None 94 95 @property 96 def available_fields(self): 97 return sorted(list(set( 98 self.location_fields + [ 99 'random_num', 'owner', 'cost', 100 'state', 'batch_serial', 101 'history',] 102 ))) 103 104 @property 105 def required_fields(self): 106 return ['random_num', 'state', 'batch_serial', 'history',] 107 108 def parentsExist(self, row, site): 109 return self.getParent(row,site) is not None 110 111 def entryExists(self, row, site): 112 parent = self.getParent(row, site) 113 if parent is None: 114 return False 115 return row['representation'] in parent.keys() 116 117 def getParent(self, row, site): 118 if not 'accesscodes' in site.keys(): 119 return None 120 batch_id = '%s-%s' % (row['batch_prefix'], row['batch_num']) 121 return site['accesscodes'].get(batch_id, None) 122 123 def getEntry(self, row, site): 124 if not self.entryExists(row, site): 125 return None 126 parent = self.getParent(row, site) 127 return parent.get(row['representation'], None) 128 129 def addEntry(self, obj, row, site): 130 parent = self.getParent(row, site) 131 obj.batch_serial = row['batch_serial'] 132 obj.random_num = row['random_num'] 133 parent[row['representation']] = obj 134 return 135 136 def updateEntry(self, obj, row, site): 137 """Update obj to the values given in row. 138 139 Returns a string describing the fields changed. 140 """ 141 items_changed = '' 142 # Update state 143 if row.has_key('state'): 144 state = row.get('state', IGNORE_MARKER) 145 if state not in (IGNORE_MARKER, ''): 146 value = row['state'] 147 IWorkflowState(obj).setState(value) 148 items_changed += ('%s=%s, ' % ('state', state)) 149 row.pop('state') 150 # Update history 151 if row.has_key('history'): 152 history = row.get('history', IGNORE_MARKER) 153 if history not in (IGNORE_MARKER, ''): 154 values = row['history'].split('||') 155 for value in values: 156 IObjectHistory(obj).addMessage(value) 157 items_changed += ('%s=%s, ' % ('history', history)) 158 row.pop('history') 159 160 # In import files we can use the hash symbol at the end of a 161 # random_num string to avoid annoying automatic number transformation 162 # by Excel or Calc 163 row['random_num'] = row['random_num'].strip('#') 164 165 items_changed += super(AccessCodeProcessor, self).updateEntry( 166 obj, row, site) 167 168 # Log actions... 169 grok.getSite()['accesscodes'].logger.info('%s - %s - AC updated: %s' 170 % (self.name, row['representation'], items_changed)) 171 return -
main/waeup.kofa/trunk/src/waeup/kofa/accesscodes/browser.py
r9217 r9265 67 67 continue 68 68 del self.context[name] 69 self.context._p_changed = True 69 70 self.flash(_('Deleted batch ${a}', mapping = {'a':name})) 70 71 message = 'deleted: %s' % name -
main/waeup.kofa/trunk/src/waeup/kofa/accesscodes/interfaces.py
r9263 r9265 49 49 owner = schema.TextLine( 50 50 title = _(u'Purchaser'), 51 required = False, 51 52 ) 52 53 history = schema.Text( … … 54 55 default = u'', 55 56 readonly = True, 57 required = False, 56 58 ) 57 59 -
main/waeup.kofa/trunk/src/waeup/kofa/accesscodes/tests/test_batching.py
r9263 r9265 36 36 from waeup.kofa.testing import ( 37 37 FunctionalLayer, FunctionalTestCase, setUp, tearDown, getRootFolder) 38 from waeup.kofa.accesscodes.batching import AccessCodeBatchProcessor 38 from waeup.kofa.accesscodes.batching import ( 39 AccessCodeBatchProcessor, AccessCodeProcessor) 39 40 40 41 … … 44 45 45 46 BATCH_HEADER_FIELDS = BATCH_SAMPLE_DATA.split( 47 '\n')[0].split(',') 48 49 AC_SAMPLE_DATA = open( 50 os.path.join(os.path.dirname(__file__), 'sample_ac_data.csv'), 51 'rb').read() 52 53 AC_HEADER_FIELDS = AC_SAMPLE_DATA.split( 46 54 '\n')[0].split(',') 47 55 … … 73 81 self.logfile = os.path.join( 74 82 self.app['datacenter'].storage, 'logs', 'accesscodes.log') 83 self.workdir = tempfile.mkdtemp() 75 84 76 85 def tearDown(self): … … 85 94 def test_import(self): 86 95 self.processor = AccessCodeBatchProcessor() 87 self.workdir = tempfile.mkdtemp()88 96 self.csv_file = os.path.join(self.workdir, 'sample_batch_data.csv') 89 97 open(self.csv_file, 'wb').write(BATCH_SAMPLE_DATA) … … 102 110 shutil.rmtree(os.path.dirname(fin_file)) 103 111 return 112 113 class ACProcessorTest(ACFullSetup): 114 115 layer = FunctionalLayer 116 117 def test_import(self): 118 self.processor = AccessCodeBatchProcessor() 119 self.csv_file = os.path.join(self.workdir, 'sample_batch_data.csv') 120 open(self.csv_file, 'wb').write(BATCH_SAMPLE_DATA) 121 self.processor.doImport(self.csv_file, BATCH_HEADER_FIELDS) 122 123 self.processor = AccessCodeProcessor() 124 self.csv_file = os.path.join(self.workdir, 'sample_ac_data.csv') 125 open(self.csv_file, 'wb').write(AC_SAMPLE_DATA) 126 num, num_warns, fin_file, fail_file = self.processor.doImport( 127 self.csv_file, AC_HEADER_FIELDS) 128 self.assertEqual(num_warns,0) 129 self.assertEqual( 130 self.app['accesscodes']['CLR-1']['CLR-1-1625368961'].batch_serial, 33) 131 self.assertEqual( 132 self.app['accesscodes']['CLR-1']['CLR-1-1625368961'].random_num, '1625368961') 133 self.assertEqual( 134 self.app['accesscodes']['CLR-1']['CLR-1-1625368961'].state, 'used') 135 logcontent = open(self.logfile).read() 136 self.assertTrue( 137 'INFO - system - AccessCode Processor - CLR-1-1625368961 - ' 138 'AC updated: state=used, history=2012-07-11 14:51:56 UTC - ' 139 'initialized by Manager||2012-07-11 14:55:42 UTC - ' 140 'invalidated by Manager, batch_serial=33, ' 141 'random_num=1625368961, cost=100.0, owner=K1000009' 142 in logcontent) 143 shutil.rmtree(os.path.dirname(fin_file)) 144 return
Note: See TracChangeset for help on using the changeset viewer.