1 # -*- encoding: utf-8 -*-
2 ##############################################################################
4 # OpenERP, Open Source Management Solution
5 # Copyright (C) 2004-2008 Tiny SPRL (<http://tiny.be>). All Rights Reserved
8 # This program is free software: you can redistribute it and/or modify
9 # it under the terms of the GNU General Public License as published by
10 # the Free Software Foundation, either version 3 of the License, or
11 # (at your option) any later version.
13 # This program is distributed in the hope that it will be useful,
14 # but WITHOUT ANY WARRANTY; without even the implied warranty of
15 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 # GNU General Public License for more details.
18 # You should have received a copy of the GNU General Public License
19 # along with this program. If not, see <http://www.gnu.org/licenses/>.
21 ##############################################################################
23 from xml.dom import minidom
24 from osv import fields,osv
30 class base_module_record(osv.osv):
31 _name = "ir.module.record"
35 def __init__(self, pool, cr=None):
36 if super(base_module_record, self).__init__.func_code.co_argcount ==3:
37 super(base_module_record, self).__init__(pool,cr)
39 super(base_module_record, self).__init__(pool)
41 self.recording_data = []
45 def _create_id(self, cr, uid, model, data):
49 name = filter(lambda x: x in string.letters, (data.get('name','') or '').lower())
52 val = model.replace('.','_')+'_'+name+ str(i)
54 if val not in self.ids.values():
58 def _get_id(self, cr, uid, model, id):
59 # if type(id)==type(()):
61 if (model,id) in self.ids:
62 return self.ids[(model,id)], False
63 dt = self.pool.get('ir.model.data')
64 dtids = dt.search(cr, uid, [('model','=',model), ('res_id','=',id)])
67 obj = dt.browse(cr, uid, dtids[0])
68 self.depends[obj.module] = True
69 return obj.module+'.'+obj.name, obj.noupdate
71 def _create_record(self, cr, uid, doc, model, data, record_id, noupdate=False):
73 if record_id not in self.ids.values():
74 record = doc.createElement('record')
75 record.setAttribute("id", record_id)
76 record.setAttribute("model", model)
77 record_list = [record]
78 lids = self.pool.get('ir.model.data').search(cr, uid, [('model','=',model)])
79 res = self.pool.get('ir.model.data').read(cr, uid, lids[:1], ['module'])
81 self.depends[res[0]['module']]=True
82 fields = self.pool.get(model).fields_get(cr, uid)
83 for key,val in data.items():
84 if not (val or (fields[key]['type']=='boolean')):
86 if fields[key]['type'] in ('integer','float'):
87 field = doc.createElement('field')
88 field.setAttribute("name", key)
89 field.setAttribute("eval", val and str(val) or 'False' )
90 record.appendChild(field)
91 elif fields[key]['type'] in ('boolean',):
92 field = doc.createElement('field')
93 field.setAttribute("name", key)
94 field.setAttribute("eval", val and '1' or '0' )
95 record.appendChild(field)
96 elif fields[key]['type'] in ('many2one',):
97 field = doc.createElement('field')
98 field.setAttribute("name", key)
99 if type(val) in (type(''),type(u'')):
102 id,update = self._get_id(cr, uid, fields[key]['relation'], val)
103 noupdate = noupdate or update
105 field.setAttribute("model", fields[key]['relation'])
106 name = self.pool.get(fields[key]['relation']).browse(cr, uid, val).name
107 if isinstance(name, basestring):
108 name = name.decode('utf8')
109 field.setAttribute("search", "[('name','=','"+name+"')]")
111 field.setAttribute("ref", id)
112 record.appendChild(field)
113 elif fields[key]['type'] in ('one2many',):
114 for valitem in (val or []):
116 if key in self.pool.get(model)._columns:
117 fname = self.pool.get(model)._columns[key]._fields_id
119 fname = self.pool.get(model)._inherit_fields[key][2]._fields_id
120 valitem[2][fname] = record_id
121 newid = self._create_id(cr, uid, fields[key]['relation'], valitem[2])
122 childrecord, update = self._create_record(cr, uid, doc, fields[key]['relation'],valitem[2], newid)
123 noupdate = noupdate or update
124 record_list += childrecord
125 self.ids[(fields[key]['relation'],newid)] = newid
128 elif fields[key]['type'] in ('many2many',):
130 for valitem in (val or []):
132 for id2 in valitem[2]:
133 id,update = self._get_id(cr, uid, fields[key]['relation'], id2)
134 self.ids[(fields[key]['relation'],id)] = id
135 noupdate = noupdate or update
137 field = doc.createElement('field')
138 field.setAttribute("name", key)
139 field.setAttribute("eval", "[(6,0,["+','.join(map(lambda x: "ref('%s')" % (x,), res))+'])]')
140 record.appendChild(field)
142 field = doc.createElement('field')
143 field.setAttribute("name", key)
145 if not isinstance(val, basestring):
148 val = val and ('"""%s"""' % val.replace('\\', '\\\\').replace('"', '\"')) or 'False'
149 if isinstance(val, basestring):
150 val=val.encode('utf8')
151 val = val.decode('utf8')
152 field.setAttribute(u"eval", val)
153 record.appendChild(field)
154 return record_list, noupdate
156 def get_o2m_list(self,cr,uid,model,id):
158 data=self.pool.get(model).read(cr, uid,id)
160 fields = self.pool.get(model).fields_get(cr, uid)
162 for key,val in data.items():
163 if fields[key]['type'] == 'many2one':
164 result[key]=data[key][0]
166 result[key]=data[key]
171 def get_copy_data(self, cr, uid,model,id):
174 data=self.pool.get(model).read(cr, uid,id)
176 fields = self.pool.get(model).fields_get(cr, uid)
178 for key,val in data.items():
179 if fields[key]['type'] == 'many2one':
180 if type(data[key])==type(True):
181 result[key]=data[key]
183 result[key]=data[key][0]
185 elif fields[key]['type'] in ('one2many',):
186 rel = fields[key]['relation']
188 for rel_id in data[key]:
189 res.append((self.get_o2m_list(cr, uid,rel,rel_id)))
192 result[key]=data[key]
194 elif fields[key]['type'] == 'many2many':
195 result[key]=[(6,0,data[key])]
198 result[key]=data[key]
199 for k,v in self.pool.get(model)._inherits.items():
203 def _generate_object_xml(self, cr, uid, rec, recv, doc, result=None):
208 id,update = self._get_id(cr, uid, rec[3], id)
209 noupdate = noupdate or update
212 record,update = self._create_record(cr, uid, doc, rec[3], rec[6], id)
213 noupdate = noupdate or update
214 record_list += record
216 elif rec[4]=='create':
217 id = self._create_id(cr, uid, rec[3],rec[5])
218 record,noupdate = self._create_record(cr, uid, doc, rec[3], rec[5], id)
219 self.ids[(rec[3],result)] = id
220 record_list += record
223 data=self.get_copy_data(cr,uid,rec[3],rec[5])
224 copy_rec=(rec[0],rec[1],rec[2],rec[3],rec[4],rec[5],data,rec[7])
226 rec_data=[(self.recording_data[0][0],rec,self.recording_data[0][2],self.recording_data[0][3])]
227 self.recording_data=rec_data
228 id = self._create_id(cr, uid, rec[3],rec[6])
229 record,noupdate = self._create_record(cr, uid, doc, rec[3], rec[6], id)
230 self.ids[(rec[3],result)] = id
231 record_list += record
233 return record_list,noupdate
235 def _generate_assert_xml(self, rec, doc):
237 def generate_xml(self, cr, uid):
238 # Create the minidom document
239 if len(self.recording_data):
241 doc = minidom.Document()
242 terp = doc.createElement("openerp")
243 doc.appendChild(terp)
244 for rec in self.recording_data:
245 if rec[0]=='workflow':
246 rec_id,noupdate = self._get_id(cr, uid, rec[1][3], rec[1][5])
249 data = doc.createElement("data")
250 terp.appendChild(data)
251 wkf = doc.createElement('workflow')
252 data.appendChild(wkf)
253 wkf.setAttribute("model", rec[1][3])
254 wkf.setAttribute("action", rec[1][4])
256 data.setAttribute("noupdate", "1")
257 wkf.setAttribute("ref", rec_id)
259 res_list,noupdate = self._generate_object_xml(cr, uid, rec[1], rec[2], doc, rec[3])
260 data = doc.createElement("data")
262 data.setAttribute("noupdate", "1")
264 terp.appendChild(data)
266 data.appendChild(res)
267 elif rec[0]=='assert':
269 res = doc.toprettyxml(indent="\t")
270 return doc.toprettyxml(indent="\t").encode('utf8')
274 def execute(*args, **argv):
275 if len(args) >= 6 and isinstance(args[5], dict):
276 _old_args = args[5].copy()
280 res = fnct(*args, **argv)
281 pool = pooler.get_pool(args[0])
282 mod = pool.get('ir.module.record')
283 if mod and mod.recording:
284 if args[4] not in ('default_get','read','fields_view_get','fields_get','search','search_count','name_search','name_get','get','request_get', 'get_sc'):
285 if _old_args is not None:
286 args[5].update(_old_args)
288 mod.recording_data.append(('query', args, argv,res))
292 def fnct_call_workflow(fnct):
293 def exec_workflow(*args, **argv):
294 res = fnct(*args, **argv)
295 pool = pooler.get_pool(args[0])
296 mod = pool.get('ir.module.record')
297 if mod and mod.recording:
298 mod.recording_data.append(('workflow', args, argv))
302 obj = netsvc._service['object']
303 obj.execute = fnct_call(obj.execute)
304 obj.exportMethod(obj.execute)
305 obj.exec_workflow = fnct_call_workflow(obj.exec_workflow)
306 obj.exportMethod(obj.exec_workflow)
308 # vim:expandtab:smartindent:tabstop=4:softtabstop=4:shiftwidth=4: