1
# -*- coding: utf-8 -*-
2
##############################################################################
4
# OpenERP, Open Source Management Solution
5
# Copyright (C) 2004-2010 Tiny SPRL (<http://tiny.be>).
7
# This program is free software: you can redistribute it and/or modify
8
# it under the terms of the GNU Affero General Public License as
9
# published by the Free Software Foundation, either version 3 of the
10
# License, or (at your option) any later version.
12
# This program is distributed in the hope that it will be useful,
13
# but WITHOUT ANY WARRANTY; without even the implied warranty of
14
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15
# GNU Affero General Public License for more details.
17
# You should have received a copy of the GNU Affero General Public License
18
# along with this program. If not, see <http://www.gnu.org/licenses/>.
20
##############################################################################
22
from xml.dom import minidom
23
from osv.osv import osv_pool
24
from osv import fields,osv
29
class recording_objects_proxy(osv_pool):
30
def execute(self, *args, **argv):
31
if args[3] == 'create':
32
_old_args = args[4].copy()
33
elif args[3] == 'write':
34
_old_args = args[5].copy()
35
elif len(args) >= 5 and isinstance(args[4], dict):
36
_old_args = args[4].copy()
37
elif len(args) > 5 and args[3] != 'write' and isinstance(args[5], dict):
38
_old_args = args[5].copy()
41
res = super(recording_objects_proxy, self).execute(*args, **argv)
42
pool = pooler.get_pool(args[0])
43
mod = pool.get('ir.module.record')
45
if mod and mod.recording:
46
if args[3] not in ('default_get','read','fields_view_get','fields_get','search','search_count','name_search','name_get','get','request_get', 'get_sc', 'unlink'):
47
if _old_args is not None:
48
if args[3] != 'write' and args[3] != 'create' and len(args) > 5 and isinstance(args[5], dict):
52
mod.recording_data.append(('osv_memory_action', args, argv ,None))
54
if args[3] == 'create':
55
args[4].update(_old_args)
56
elif args[3] == 'write':
57
args[5].update(_old_args)
58
mod.recording_data.append(('query', args, argv,res))
61
def exec_workflow(self, *args, **argv):
62
res = super(recording_objects_proxy, self).exec_workflow(*args, **argv)
63
pool = pooler.get_pool(args[0])
64
mod = pool.get('ir.module.record')
65
if mod and mod.recording:
66
mod.recording_data.append(('workflow', args, argv))
69
recording_objects_proxy()
71
class xElement(minidom.Element):
72
"""dom.Element with compact print
73
The Element in minidom has a problem: if printed, adds whitespace
74
around the text nodes. The standard will not ignore that whitespace.
75
This class simply prints the contained nodes in their compact form, w/o
78
def writexml(self, writer, indent="", addindent="", newl=""):
80
minidom.Element.writexml(self, writer, indent='', addindent='', newl='')
83
def doc_createXElement(xdoc, tagName):
85
e.ownerDocument = xdoc
89
from tools import yaml_tag # This import is not unused! Do not remove!
90
# Please do not override yaml_tag here: modify it in server bin/tools/yaml_tag.py
92
class base_module_record(osv.osv):
93
_name = "ir.module.record"
97
def __init__(self, *args, **kwargs):
99
self.recording_data = []
101
super(base_module_record, self).__init__(*args, **kwargs)
104
def _create_id(self, cr, uid, model, data):
108
name = filter(lambda x: x in string.letters, (data.get('name','') or '').lower())
111
# name=data.get('name','') or ''.lower()
112
val = model.replace('.','_')+'_'+ name + str(i)
114
if val not in self.ids.values():
118
def _get_id(self, cr, uid, model, id):
119
if type(id)==type(()):
121
if (model,id) in self.ids:
122
res_id = self.ids[(model,id)]
124
dt = self.pool.get('ir.model.data')
125
dtids = dt.search(cr, uid, [('model','=',model), ('res_id','=',id)])
128
obj = dt.browse(cr, uid, dtids[0])
129
self.depends[obj.module] = True
130
return obj.module+'.'+obj.name, obj.noupdate
132
def _create_record(self, cr, uid, doc, model, data, record_id, noupdate=False):
133
data_pool = self.pool.get('ir.model.data')
134
model_pool = self.pool.get(model)
136
record = doc.createElement('record')
137
record.setAttribute("id", record_id)
138
record.setAttribute("model", model)
139
record_list = [record]
141
lids = data_pool.search(cr, uid, [('model','=',model)])
142
res = data_pool.read(cr, uid, lids[:1], ['module'])
144
self.depends[res[0]['module']]=True
145
fields = model_pool.fields_get(cr, uid)
146
for key,val in data.items():
147
if not (val or (fields[key]['type']=='boolean')):
149
if fields[key]['type'] in ('integer','float'):
150
field = doc.createElement('field')
151
field.setAttribute("name", key)
152
field.setAttribute("eval", val and str(val) or 'False' )
153
record.appendChild(field)
154
elif fields[key]['type'] in ('boolean',):
155
field = doc.createElement('field')
156
field.setAttribute("name", key)
157
field.setAttribute("eval", val and '1' or '0' )
158
record.appendChild(field)
159
elif fields[key]['type'] in ('many2one',):
160
field = doc.createElement('field')
161
field.setAttribute("name", key)
162
if type(val) in (type(''),type(u'')):
165
id,update = self._get_id(cr, uid, fields[key]['relation'], val)
166
noupdate = noupdate or update
168
relation_pool = self.pool.get(fields[key]['relation'])
170
field.setAttribute("model", fields[key]['relation'])
171
fld_nm = relation_pool._rec_name
172
name = relation_pool.read(cr, uid, val,[fld_nm])[fld_nm] or False
173
field.setAttribute("search", str([(str(fld_nm) ,'=', name)]))
175
field.setAttribute("ref", id)
176
record.appendChild(field)
177
elif fields[key]['type'] in ('one2many',):
178
for valitem in (val or []):
179
if valitem[0] in (0,1):
180
if key in model_pool._columns:
181
model_pool._columns[key]._fields_id
183
model_pool._inherit_fields[key][2]._fields_id
185
newid = self._create_id(cr, uid, fields[key]['relation'], valitem[2])
188
newid,update = self._get_id(cr, uid, fields[key]['relation'], valitem[1])
190
newid = self._create_id(cr, uid, fields[key]['relation'], valitem[2])
192
self.ids[(fields[key]['relation'], valitem[1])] = newid
194
childrecord, update = self._create_record(cr, uid, doc, fields[key]['relation'],valitem[2], newid)
195
noupdate = noupdate or update
196
record_list += childrecord
199
elif fields[key]['type'] in ('many2many',):
201
for valitem in (val or []):
203
for id2 in valitem[2]:
204
id,update = self._get_id(cr, uid, fields[key]['relation'], id2)
205
self.ids[(fields[key]['relation'],id2)] = id
206
noupdate = noupdate or update
208
field = doc.createElement('field')
209
field.setAttribute("name", key)
210
field.setAttribute("eval", "[(6,0,["+','.join(map(lambda x: "ref('%s')" % (x,), res))+'])]')
211
record.appendChild(field)
213
field = doc_createXElement(doc, 'field')
214
field.setAttribute("name", key)
215
field.appendChild(doc.createTextNode(val))
216
record.appendChild(field)
218
return record_list, noupdate
220
def _create_yaml_record(self, cr, uid, model, data, record_id):
221
record={'model': model, 'id': str(record_id)}
223
model_pool = self.pool.get(model)
224
data_pool = self.pool.get('ir.model.data')
225
lids = data_pool.search(cr, uid, [('model','=',model)])
227
res = data_pool.read(cr, uid, lids[:1], ['module'])
230
self.depends[res[0]['module']]=True
231
fields = model_pool.fields_get(cr, uid)
234
defaults[model] = model_pool.default_get(cr, uid, data)
237
for key,val in data.items():
238
if ((key in defaults[model]) and (val == defaults[model][key])) and not(fields[key].get('required',False)):
240
if fields[key]['type'] in ('integer','float'):
244
elif not (val or (fields[key]['type']=='function')):
246
elif fields[key]['type'] in ('boolean',):
250
elif fields[key]['type'] in ('many2one',):
251
if type(val) in (type(''), type(u'')):
254
id, update = self._get_id(cr, uid, fields[key]['relation'], val)
256
elif fields[key]['type'] in ('one2many',):
258
for valitem in (val or []):
259
if valitem[0] in (0,1):
260
if key in model_pool._columns:
261
fname = model_pool._columns[key]._fields_id
263
fname = model_pool._inherit_fields[key][2]._fields_id
264
del valitem[2][fname] #delete parent_field from child's fields list
266
childrecord = self._create_yaml_record(cr, uid, fields[key]['relation'],valitem[2], None)
267
items[0].append(childrecord['attrs'])
269
elif fields[key]['type'] in ('many2many',):
270
if (key in defaults[model]) and (val[0][2] == defaults[model][key]):
273
for valitem in (val or []):
275
for id2 in valitem[2]:
276
id,update = self._get_id(cr, uid, fields[key]['relation'], id2)
277
self.ids[(fields[key]['relation'],id2)] = id
286
attrs[key]=tools.ustr(val)
287
attrs[key]=attrs[key].replace('"','\'')
288
record['attrs'] = attrs
291
def get_copy_data(self, cr, uid, model, id, result):
293
obj=self.pool.get(model)
294
data=obj.read(cr, uid,[id])
295
if type(data)==type([]):
301
mod_fields = obj.fields_get(cr, uid)
302
for f in filter(lambda a: isinstance(obj._columns[a], fields.function)\
303
and (not obj._columns[a].store),obj._columns):
306
for key,val in data.items():
307
if result.has_key(key):
309
if mod_fields[key]['type'] == 'many2one':
310
if type(data[key])==type(True) or type(data[key])==type(1):
311
result[key]=data[key]
315
result[key]=data[key][0]
317
elif mod_fields[key]['type'] in ('one2many',):
318
# continue # due to this start stop recording will not record one2many field
319
rel = mod_fields[key]['relation']
322
for rel_id in data[key]:
324
res.append(self.get_copy_data(cr, uid,rel,rel_id,{}))
328
result[key]=data[key]
330
elif mod_fields[key]['type'] == 'many2many':
331
result[key]=[(6,0,data[key])]
334
result[key]=data[key]
335
for k,v in obj._inherits.items():
339
def _create_function(self, cr, uid, doc, model, name, record_id):
340
record = doc.createElement('function')
341
record.setAttribute("name", name)
342
record.setAttribute("model", model)
343
record_list = [record]
345
value = doc.createElement('value')
346
value.setAttribute('eval', '[ref(\'%s\')]' % (record_id, ))
347
value.setAttribute('model', model)
349
record.appendChild(value)
350
return record_list, False
352
def _generate_object_xml(self, cr, uid, rec, recv, doc, result=None):
357
id,update = self._get_id(cr, uid, rec[2], id)
358
noupdate = noupdate or update
361
record,update = self._create_record(cr, uid, doc, rec[2], rec[5], id)
362
noupdate = noupdate or update
363
record_list += record
365
elif rec[4] in ('menu_create',):
367
id,update = self._get_id(cr, uid, rec[3], id)
368
noupdate = noupdate or update
371
record,update = self._create_function(cr, uid, doc, rec[3], rec[4], id)
372
noupdate = noupdate or update
373
record_list += record
375
elif rec[3]=='create':
376
id = self._create_id(cr, uid, rec[2],rec[4])
377
record,noupdate = self._create_record(cr, uid, doc, rec[2], rec[4], id)
378
self.ids[(rec[2], result)] = id
379
record_list += record
382
data=self.get_copy_data(cr,uid,rec[2],rec[4],rec[5])
383
copy_rec=(rec[0],rec[1],rec[2],rec[3],rec[4],data,rec[5])
385
rec_data=[(self.recording_data[0][0],rec,self.recording_data[0][2],self.recording_data[0][3])]
386
self.recording_data=rec_data
387
id = self._create_id(cr, uid, rec[2],rec[5])
388
record,noupdate = self._create_record(cr, uid, doc, rec[2], rec[5], id)
389
self.ids[(rec[2], result)] = id
390
record_list += record
392
return record_list,noupdate
394
def _generate_object_yaml(self, cr, uid, rec, result=None):
395
if self.mode=="create":
396
yml_id = self._create_id(cr, uid, rec[2],rec[4])
397
self.ids[(rec[2], result)] = yml_id
398
record = self._create_yaml_record(cr, uid, rec[2], rec[4], yml_id)
400
if self.mode=="workflow":
401
id,update = self._get_id(cr, uid, rec[2], rec[4])
403
data['model'] = rec[2]
404
data['action'] = rec[3]
407
if self.mode=="write":
408
id,update = self._get_id(cr, uid, rec[2],rec[4][0])
409
record = self._create_yaml_record(cr, uid, rec[2], rec[5], id)
411
data=self.get_copy_data(cr,uid,rec[2],rec[4],rec[5])
412
copy_rec=(rec[0],rec[1],rec[2],rec[3],rec[4],data,rec[5])
414
rec_data=[(self.recording_data[0][0],rec,self.recording_data[0][2],self.recording_data[0][3])]
415
self.recording_data=rec_data
416
id = self._create_id(cr, uid, rec[2],rec[5])
417
record = self._create_yaml_record(cr, uid, str(rec[2]), rec[5], id)
418
self.ids[(rec[2], result)] = id
421
def _generate_function_yaml(self, cr, uid, args):
422
db, uid, model, action, ids, context = args
423
temp_context = context.copy()
424
active_id = temp_context['active_id']
425
active_model = temp_context['active_model']
426
active_id, update = self._get_id(cr, uid, active_model, active_id)
429
rec_id, noupdate = self._get_id(cr, uid, model, ids[0])
430
temp_context['active_id'] = "ref('%s')"%unicode(active_id)
431
temp_context['active_ids'][0] = "ref('%s')"%str(active_id)
433
function['model'] = model
434
function['action'] = action
435
attrs = "self.%s(cr, uid, [ref('%s')], {" %(action, rec_id, )
436
for k, v in temp_context.iteritems():
437
if isinstance(v, str):
438
f= "'"+k+"': "+"'%s'"%v + ", "
440
v=str(v).replace('"', '')
441
f= "'"+k+"': "+"%s"%v + ", "
443
attrs=str(attrs)+'})'
444
function['attrs'] = attrs
447
def _generate_assert_xml(self, rec, doc):
450
def generate_xml(self, cr, uid):
451
# Create the minidom document
452
if len(self.recording_data):
454
doc = minidom.Document()
455
terp = doc.createElement("openerp")
456
doc.appendChild(terp)
457
for rec in self.recording_data:
458
if rec[0]=='workflow':
459
rec_id,noupdate = self._get_id(cr, uid, rec[1][2], rec[1][4])
462
data = doc.createElement("data")
463
terp.appendChild(data)
464
wkf = doc.createElement('workflow')
465
data.appendChild(wkf)
466
wkf.setAttribute("model", rec[1][2])
467
wkf.setAttribute("action", rec[1][3])
469
data.setAttribute("noupdate", "1")
470
wkf.setAttribute("ref", rec_id)
472
res_list,noupdate = self._generate_object_xml(cr, uid, rec[1], rec[2], doc, rec[3])
473
data = doc.createElement("data")
475
data.setAttribute("noupdate", "1")
477
terp.appendChild(data)
479
data.appendChild(res)
480
elif rec[0]=='assert':
482
return doc.toprettyxml(indent="\t").encode('utf-8')
484
def generate_yaml(self, cr, uid):
486
if len(self.recording_data):
489
for rec in self.recording_data:
490
if rec[1][3] == 'create':
492
elif rec[1][3] == 'write':
494
elif rec[1][3] == 'copy':
496
elif rec[0] == 'workflow':
498
elif rec[0] == 'osv_memory_action':
499
self.mode='osv_memory_action'
502
if self.mode == "workflow":
503
record = self._generate_object_yaml(cr, uid, rec[1],rec[0])
504
yaml_file += "!comment Performing a workflow action %s on module %s"%(record['action'], record['model']) + '''\n'''
505
object = yaml.load(unicode('''\n !workflow %s \n'''%record,'iso-8859-1'))
506
yaml_file += str(object) + '''\n\n'''
507
elif self.mode == 'osv_memory_action':
508
osv_action = self._generate_function_yaml(cr, uid, rec[1])
509
yaml_file += "!comment Performing an osv_memory action %s on module %s"%(osv_action['action'], osv_action['model']) + '''\n'''
510
osv_action = yaml.load(unicode('''\n !python %s \n'''%osv_action,'iso-8859-1'))
511
yaml_file += str(osv_action) + '''\n'''
512
attrs = yaml.dump(osv_action.attrs, default_flow_style=False)
513
attrs = attrs.replace("''", '"')
514
attrs = attrs.replace("'", '')
515
yaml_file += attrs + '''\n\n'''
517
record = self._generate_object_yaml(cr, uid, rec[1], rec[3])
518
if self.mode == "create" or self.mode == "copy":
519
yaml_file += "!comment Creating a %s record"%(record['model']) + '''\n'''
521
yaml_file += "!comment Modifying a %s record"%(record['model']) + '''\n'''
522
object = yaml.load(unicode('''\n !record %s \n'''%record,'iso-8859-1'))
523
yaml_file += str(object) + '''\n'''
524
attrs = yaml.dump(object.attrs, default_flow_style=False)
525
yaml_file += attrs + '''\n\n'''
528
for line in yaml_file.split('\n'):
529
line=line.replace("''","'")
530
if (line.find('!record') == 0) or (line.find('!workflow') == 0) or (line.find('!python') == 0):
531
line = "- \n" + " " + line
532
elif line.find('!comment') == 0:
533
line=line.replace('!comment','- \n ')
534
elif line.find('- -') != -1:
535
line=line.replace('- -',' -')
539
yaml_result += line + '''\n'''
544
# vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4: